openFPGALoader/bitparser.cpp

146 lines
3.2 KiB
C++
Raw Normal View History

2019-09-26 18:29:20 +02:00
#include "bitparser.hpp"
#include <stdio.h>
#include <stdlib.h>
#include <iostream>
#include <arpa/inet.h>
using namespace std;
#ifdef DEBUG
#define display(...) \
do { if (1) fprintf(stdout, __VA_ARGS__);} while(0)
#else
#define display(...) do {} while(0)
#endif
BitParser::BitParser(string filename):
ConfigBitstreamParser(filename, ConfigBitstreamParser::BIN_MODE),
2019-09-26 18:29:20 +02:00
fieldA(), part_name(), date(), hour(),
design_name(), userID(), toolVersion()
2019-09-26 18:29:20 +02:00
{
}
BitParser::~BitParser()
{
}
int BitParser::parseField()
2019-09-26 18:29:20 +02:00
{
int ret = 1;
2019-09-26 18:29:20 +02:00
short length;
char tmp[64];
int pos, prev_pos;
/* type */
uint8_t type;
_fd.read((char *)&type, sizeof(uint8_t));
2019-09-26 18:29:20 +02:00
if (type != 'e') {
_fd.read((char*)&length, sizeof(uint16_t));
2019-09-26 18:29:20 +02:00
length = ntohs(length);
} else {
length = 4;
}
_fd.read(tmp, sizeof(uint8_t)*length);
2019-09-26 18:29:20 +02:00
#ifdef DEBUG
for (int i = 0; i < length; i++)
printf("%c", tmp[i]);
printf("\n");
#endif
switch (type) {
case 'a': /* design name:userid:synthesize tool version */
fieldA=(tmp);
prev_pos = 0;
pos = fieldA.find(";");
design_name = fieldA.substr(prev_pos, pos);
printf("%d %d %s\n", prev_pos, pos, design_name.c_str());
prev_pos = pos+1;
pos = fieldA.find(";", prev_pos);
userID = fieldA.substr(prev_pos, pos-prev_pos);
printf("%d %d %s\n", prev_pos, pos, userID.c_str());
prev_pos = pos+1;
//pos = fieldA.find(";", prev_pos);
toolVersion = fieldA.substr(prev_pos);
printf("%d %d %s\n", prev_pos, pos, toolVersion.c_str());
break;
case 'b': /* FPGA model */
part_name = (tmp);
break;
case 'c': /* buildDate */
date = (tmp);
break;
case 'd': /* buildHour */
hour = (tmp);
break;
case 'e': /* file size */
_bit_length = 0;
2019-09-26 18:29:20 +02:00
for (int i = 0; i < 4; i++) {
#ifdef DEBUG
printf("%x %x\n", 0xff & tmp[i], _bit_length);
2019-09-26 18:29:20 +02:00
#endif
_bit_length <<= 8;
_bit_length |= 0xff & tmp[i];
2019-09-26 18:29:20 +02:00
}
#ifdef DEBUG
printf(" %x\n", _bit_length);
2019-09-26 18:29:20 +02:00
#endif
ret = 0;
2019-09-26 18:29:20 +02:00
break;
}
return ret;
2019-09-26 18:29:20 +02:00
}
int BitParser::parse()
{
uint16_t length;
2019-09-26 18:29:20 +02:00
display("parser\n\n");
/* Field 1 : misc header */
_fd.read((char*)&length, sizeof(uint16_t));
2019-09-26 18:29:20 +02:00
length = ntohs(length);
_fd.seekg(length, _fd.cur);
_fd.read((char*)&length, sizeof(uint16_t));
2019-09-26 18:29:20 +02:00
length = ntohs(length);
/* process all field */
do {} while (parseField());
2019-09-26 18:29:20 +02:00
display("results\n\n");
cout << "fieldA : " << fieldA << endl;
cout << " : " << design_name << ";" << userID << ";" << toolVersion << endl;
cout << "part name : " << part_name << endl;
cout << "date : " << date << endl;
cout << "hour : " << hour << endl;
cout << "file length : " << _bit_length << endl;
2019-09-26 18:29:20 +02:00
/* rest of the file is data to send */
int pos = _fd.tellg();
cout << pos + _bit_length << endl;
_fd.read((char *)&_bit_data[0], sizeof(uint8_t) * _bit_length);
if (_fd.gcount() != _bit_length) {
cerr << "Error: data read different to asked length ";
cerr << to_string(_fd.gcount()) << " " << to_string(_bit_length) << endl;
2019-09-26 18:29:20 +02:00
return -1;
}
for (int i = 0; i < _bit_length; i++) {
_bit_data[i] = reverseByte(_bit_data[i]);
2019-09-26 18:29:20 +02:00
}
return 0;
}
uint8_t BitParser::reverseByte(uint8_t src)
2019-09-26 18:29:20 +02:00
{
uint8_t dst = 0;
2019-09-26 18:29:20 +02:00
for (int i=0; i < 8; i++) {
dst = (dst << 1) | (src & 0x01);
src >>= 1;
}
return dst;
}