/* Copyright (C) 2009 Her Majesty the Queen in Right of Canada (Communications Research Center Canada) */ /* This file is part of ODR-DabMux. ODR-DabMux is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. ODR-DabMux is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with ODR-DabMux. If not, see <http://www.gnu.org/licenses/>. */ #include "dabInputDabplusFile.h" #include "dabInput.h" #include <stdio.h> #include <string.h> #include <fcntl.h> #ifndef _WIN32 # define O_BINARY 0 #endif #ifdef HAVE_FORMAT_DABPLUS # ifdef HAVE_INPUT_FILE struct dabInputOperations dabInputDabplusFileOperations = { dabInputDabplusFileInit, dabInputDabplusFileOpen, dabInputSetbuf, dabInputDabplusFileRead, nullptr, nullptr, dabInputDabplusFileReadFrame, dabInputSetbitrate, dabInputDabplusFileClose, dabInputDabplusFileClean, dabInputDabplusFileRewind }; int dabInputDabplusFileInit(void** args) { dabInputDabplusFileData* data = new dabInputDabplusFileData; data->file = -1; data->buffer = nullptr; data->bufferSize = 0; data->bufferIndex = 0; *args = data; return 0; } int dabInputDabplusFileOpen(void* args, const char* filename) { dabInputDabplusFileData* data = (dabInputDabplusFileData*)args; data->file = open(filename, O_RDONLY | O_BINARY); if (data->file == -1) { perror(filename); return -1; } return 0; } int dabInputDabplusFileRead(void* args, void* buffer, int size) { dabInputDabplusFileData* data = (dabInputDabplusFileData*)args; if (data->bufferSize != (size_t)size * 5) { if (data->buffer == nullptr) { delete[] data->buffer; } data->buffer = new uint8_t[size * 5]; memset(data->buffer, 0, size * 5); data->bufferSize = size * 5; data->bufferIndex = 0; } if (data->bufferIndex + size > data->bufferSize) { int ret = read(data->file, data->buffer, data->bufferSize); if (ret != (int)data->bufferSize) { if (ret != 0) { etiLog.log(alert, "ERROR: Incomplete DAB+ frame!\n"); } return 0; } data->bufferIndex = 0; } memcpy(buffer, &data->buffer[data->bufferIndex], size); data->bufferIndex += size; return size; } int dabInputDabplusFileReadFrame(dabInputOperations* ops, void* args, void* buffer, int size) { //dabInputDabplusFileData* data = (dabInputDabplusFileData*)args; int result; uint8_t* dataOut = reinterpret_cast<uint8_t*>(buffer); result = ops->read(args, dataOut, size); if (result == -1) { etiLog.log(alert, "ERROR: Can't read file\n"); perror(""); return -1; } if (result < size) { int sizeOut = result; etiLog.log(info, "reach end of file -> rewinding\n"); if (ops->rewind(args) == -1) { etiLog.log(alert, "ERROR: Can't rewind file\n"); return -1; } result = ops->read(args, dataOut + sizeOut, size - sizeOut); if (result == -1) { etiLog.log(alert, "ERROR: Can't read file\n"); perror(""); return -1; } if (result < size) { etiLog.log(alert, "ERROR: Not enought data in file\n"); return -1; } } return size; } int dabInputDabplusFileClose(void* args) { dabInputDabplusFileData* data = (dabInputDabplusFileData*)args; if (data->file != -1) { close(data->file); } return 0; } int dabInputDabplusFileClean(void** args) { dabInputDabplusFileData* data = (dabInputDabplusFileData*)*args; if (data->buffer != nullptr) { delete[] data->buffer; } delete data; return 0; } int dabInputDabplusFileRewind(void* args) { dabInputDabplusFileData* data = (dabInputDabplusFileData*)args; return lseek(data->file, 0, SEEK_SET); } # endif #endif