LibUSB - Потоковое видео на микроконтроллер заикается (LibUSBK Bulk Mode)
Я пытаюсь передать видеофайл в MCU, но всякий раз, когда он воспроизводит видео на устройстве, он довольно умеренно заикается. Если я использую свой компьютер во время потоковой передачи, ситуация становится значительно хуже.
Например, если я веду потоковую передачу, а затем просматриваю историю чата, скажем, в Discord, заикание становится намного, намного хуже, что невероятно странно, но может быть ключевым фактором в поиске корня проблемы.
ПРИМЕЧАНИЕ. Мы должны передавать 2352 байта фрагментами по 512 байтов (это для формата видео диска ISO9660, который я транслирую, и критически важен для системы, которая его воспроизводит, поэтому не думайте, что это проблема или ошибка).
Вот моя текущая реализация кода.
// main streaming loop
// we send the data by 512 bytes, so 2352 fits into 5 x 512
// [0] code
// [1] sync_id
// [2] readsync_id
// [3, 4, 5, 6] cd_lba
int VideoStream(char *image_path)
{
#define BYTES_PER_SECTOR 2352
int error = 0; // the error returned from the LibUSB transfer functions
int received = 0;
int length = 0;
unsigned char curr_req_id = 0, req_id = 0;
unsigned char curr_rdsync_id = 0, rdsync_id = 0;
unsigned char wr_lock = 0;
unsigned char *pbuf;
int fread_result; // contains the result from 'fread'
unsigned char code;
unsigned int cd_lba, offset;
memset(cd_buffer, 0, CD_BUFFER_MAX_LEN); // clear the 'cd_buffer' array before we use it
fp = fopen(image_path, "rb"); // open the video file (read mode)
if (fp == NULL)
{
#ifdef DEBUG
printf("Failed to open the video file \"%s\"\n", image_path);
#endif
return 0;
}
fseek(fp, 0L, SEEK_END);
// calculate the size of the file
size_t fsize = ftell(fp);
if (fsize <= 0) return 0;
fseek(fp, 0, SEEK_SET);
// clear the 'cd_buffer' array again
memset(cd_buffer, 0, CD_BUFFER_MAX_LEN);
// main streaming loop
while (1)
{
if(libusb_interrupt_transfer(handle, BULK_EP_IN, cd_buffer, 16, &transferred, 3000) < 0)
{
StopUSBStream();
// these crash the program? why??
//libusb_close(handle);
//libusb_exit(NULL);
return 0;
}
code = cd_buffer[0];
curr_req_id = cd_buffer[1];
curr_rdsync_id = cd_buffer[2];
if (req_id == curr_req_id) continue;
req_id = curr_req_id;
if (code == 0x99)
{
// idle
}
else if (code == 0x01) // data request
{
if (wr_lock)
{
if (rdsync_id != curr_rdsync_id)
{
wr_lock = 0;
}
}
if (!wr_lock)
{
cd_lba = cd_buffer[3] + 0x0000100 * cd_buffer[4] + 0x0010000 * cd_buffer[5] + 0x1000000 * cd_buffer[6];
offset = (cd_lba - 150) * 0x930;
fseek(fp, 0, SEEK_SET);
fseek(fp, offset, SEEK_SET);
fread_result = fread(cd_buffer + 4, 1, BYTES_PER_SECTOR, fp);
if (fread_result < BYTES_PER_SECTOR)
{
int rd_try = 10;
int rd_cmplt = 0;
int rd_more = BYTES_PER_SECTOR;
int rd_total = fread_result;
while (rd_try)
{
rd_more -= fread_result;
offset += fread_result;
fseek(fp, offset, SEEK_SET);
fread_result = fread(cd_buffer + 4, 1, rd_more, fp);
rd_total += fread_result;
if (rd_total == BYTES_PER_SECTOR) rd_cmplt = 1; // if we have read all the bytes, flag that the reading is completed
rd_try--; // negate 1 byte
}
// if reading is still not complete, flag that we have an error
if (!rd_cmplt) rd_cmplt = 0;
}
pbuf = &cd_buffer[2];
int n = CD_BUFFER_MAX_LEN;
int c = 0;
int wrlen = 512;
while (n)
{
pbuf[0] = 2;
pbuf[1] = c;
error = libusb_bulk_transfer(handle, BULK_EP_OUT, pbuf, wrlen, &transferred, 0);
pbuf += (wrlen-2);
n -= wrlen;
c++;
}
if (error == 0)
{
rdsync_id = curr_rdsync_id;
wr_lock = 1;
}
}
}
}
}
Любые предложения относительно того, почему это может происходить? Мы используем
Любая помощь будет принята с благодарностью.