2022-01-07 12:17:44 -06:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
2022-08-25 18:01:39 -07:00
|
|
|
// SCSI Target Emulator RaSCSI Reloaded
|
2022-01-07 12:17:44 -06:00
|
|
|
// for Raspberry Pi
|
|
|
|
//
|
|
|
|
// Powered by XM6 TypeG Technology.
|
|
|
|
// Copyright (C) 2016-2020 GIMONS
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2022-11-10 07:44:06 +01:00
|
|
|
#include "shared/log.h"
|
|
|
|
#include "shared/rascsi_version.h"
|
|
|
|
#include "sm_reports.h"
|
2022-01-07 12:17:44 -06:00
|
|
|
#include <iostream>
|
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
const static string html_header = R"(
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<style>
|
|
|
|
table, th, td {
|
|
|
|
border: 1px solid black;
|
|
|
|
border-collapse: collapse;
|
|
|
|
}
|
|
|
|
h1 {
|
|
|
|
text-align: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
h2 {
|
|
|
|
text-align: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
h3 {
|
|
|
|
text-align: center;
|
|
|
|
}
|
|
|
|
pre {
|
|
|
|
text-align: center;
|
|
|
|
}
|
|
|
|
.collapsible {
|
|
|
|
background-color: #777;
|
|
|
|
color: white;
|
|
|
|
cursor: pointer;
|
|
|
|
width: 100%;
|
|
|
|
border: none;
|
|
|
|
text-align: left;
|
|
|
|
outline: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
.active, .collapsible:hover {
|
|
|
|
background-color: #555;
|
|
|
|
}
|
|
|
|
|
|
|
|
.content {
|
|
|
|
padding: 0;
|
|
|
|
display: none;
|
|
|
|
overflow: hidden;
|
|
|
|
background-color: #f1f1f1;
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
</head>
|
|
|
|
)";
|
|
|
|
|
|
|
|
static void print_copyright_info(ofstream& html_fp)
|
|
|
|
{
|
|
|
|
html_fp << "<table>" << endl \
|
|
|
|
<< "<h1>RaSCSI scsimon Capture Tool</h1>" << endl \
|
|
|
|
<< "<pre>Version " << rascsi_get_version_string() \
|
|
|
|
<< __DATE__ << " " << __TIME__ << endl \
|
|
|
|
<< "Copyright (C) 2016-2020 GIMONS" << endl \
|
|
|
|
<< "Copyright (C) 2020-2021 Contributors to the RaSCSI project" << endl \
|
|
|
|
<< "</pre></table>" << endl \
|
|
|
|
<< "<br>" << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const string html_footer = R"(
|
|
|
|
</table>
|
|
|
|
<script>
|
|
|
|
var coll = document.getElementsByClassName("collapsible");
|
|
|
|
var i;
|
|
|
|
|
|
|
|
for (i = 0; i < coll.length; i++) {
|
|
|
|
coll[i].addEventListener("click", function() {
|
|
|
|
this.classList.toggle("active");
|
|
|
|
var content = this.nextElementSibling;
|
|
|
|
if (content.style.display === "block") {
|
|
|
|
content.style.display = "none";
|
|
|
|
} else {
|
|
|
|
content.style.display = "block";
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
</html>
|
|
|
|
)";
|
|
|
|
|
|
|
|
|
2022-10-08 19:26:04 +02:00
|
|
|
static void print_html_data(ofstream& html_fp, const data_capture *data_capture_array, uint32_t capture_count)
|
2022-01-07 12:17:44 -06:00
|
|
|
{
|
|
|
|
const data_capture *data;
|
|
|
|
bool prev_data_valid = false;
|
|
|
|
bool curr_data_valid;
|
2022-10-08 19:26:04 +02:00
|
|
|
uint32_t selected_id = 0;
|
2022-09-21 08:27:51 +02:00
|
|
|
BUS::phase_t prev_phase = BUS::phase_t::busfree;
|
2022-01-07 12:17:44 -06:00
|
|
|
bool close_row = false;
|
|
|
|
int data_space_count = 0;
|
|
|
|
bool collapsible_div_active = false;
|
|
|
|
bool button_active = false;
|
|
|
|
|
|
|
|
html_fp << "<table>" << endl;
|
|
|
|
|
2022-10-08 19:26:04 +02:00
|
|
|
for (uint32_t idx = 0; idx < capture_count; idx++) {
|
2022-01-07 12:17:44 -06:00
|
|
|
data = &data_capture_array[idx];
|
|
|
|
curr_data_valid = GetAck(data) && GetReq(data);
|
|
|
|
BUS::phase_t phase = GetPhase(data);
|
2022-10-08 19:26:04 +02:00
|
|
|
if (phase == BUS::phase_t::selection && !GetBsy(data)) {
|
2022-01-07 12:17:44 -06:00
|
|
|
selected_id = GetData(data);
|
|
|
|
}
|
2022-10-08 19:26:04 +02:00
|
|
|
if (prev_phase != phase) {
|
|
|
|
if (close_row) {
|
|
|
|
if (collapsible_div_active) {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << "</code></div>";
|
|
|
|
}
|
2022-10-08 19:26:04 +02:00
|
|
|
else if (button_active) {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << "</code></button>";
|
|
|
|
}
|
|
|
|
html_fp << "</td>";
|
2022-10-08 19:26:04 +02:00
|
|
|
if (data_space_count < 1) {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << "<td>--</td>";
|
|
|
|
}
|
2022-10-08 19:26:04 +02:00
|
|
|
else {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << "<td>wc: " << std::dec << "(0x" << std::hex << data_space_count << ")</td>";
|
|
|
|
}
|
|
|
|
html_fp << "</tr>" << endl;
|
|
|
|
data_space_count = 0;
|
|
|
|
}
|
|
|
|
html_fp << "<tr>";
|
|
|
|
close_row = true; // Close the row the next time around
|
|
|
|
html_fp << "<td>" << (double)data->timestamp / 100000 << "</td>";
|
|
|
|
html_fp << "<td>" << GetPhaseStr(data) << "</td>";
|
|
|
|
html_fp << "<td>" << std::hex << selected_id << "</td>";
|
|
|
|
html_fp << "<td>";
|
|
|
|
}
|
2022-10-08 19:26:04 +02:00
|
|
|
if (curr_data_valid && !prev_data_valid) {
|
|
|
|
if (data_space_count == 0) {
|
2022-01-07 12:17:44 -06:00
|
|
|
button_active = true;
|
|
|
|
html_fp << "<button type=\"button\" class=\"collapsible\"><code>";
|
|
|
|
}
|
2022-10-08 19:26:04 +02:00
|
|
|
if ((data_space_count % 16) == 0) {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << std::hex << data_space_count << ": ";
|
|
|
|
}
|
|
|
|
|
|
|
|
html_fp << fmt::format("{0:02X}", GetData(data));
|
|
|
|
|
|
|
|
data_space_count++;
|
2022-10-08 19:26:04 +02:00
|
|
|
if ((data_space_count % 4) == 0) {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << " ";
|
|
|
|
}
|
2022-10-08 19:26:04 +02:00
|
|
|
if (data_space_count == 16) {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << "</code></button><div class=\"content\"><code>" << endl;
|
|
|
|
collapsible_div_active = true;
|
|
|
|
button_active = false;
|
|
|
|
}
|
2022-10-08 19:26:04 +02:00
|
|
|
if (((data_space_count % 16) == 0) && (data_space_count > 17)) {
|
2022-01-07 12:17:44 -06:00
|
|
|
html_fp << "<br>" << endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
prev_data_valid = curr_data_valid;
|
|
|
|
prev_phase = phase;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-08 19:26:04 +02:00
|
|
|
void scsimon_generate_html(const char *filename, const data_capture *data_capture_array, uint32_t capture_count)
|
2022-01-07 12:17:44 -06:00
|
|
|
{
|
2022-10-08 19:26:04 +02:00
|
|
|
LOGINFO("Creating HTML report file (%s)", filename)
|
2022-01-07 12:17:44 -06:00
|
|
|
|
|
|
|
ofstream html_ofstream;
|
|
|
|
|
|
|
|
html_ofstream.open(filename, ios::out);
|
|
|
|
|
|
|
|
html_ofstream << html_header;
|
|
|
|
print_copyright_info(html_ofstream);
|
|
|
|
print_html_data(html_ofstream, data_capture_array, capture_count);
|
|
|
|
|
|
|
|
html_ofstream << html_footer;
|
|
|
|
|
|
|
|
html_ofstream.close();
|
|
|
|
}
|