#include "programFpgaBlackfin.h" #include "clogger.h" #include "common.h" #include "sls/ansi.h" #include "slsDetectorServer_defs.h" #include #include // usleep #include /* global variables */ // clang-format off #define MAX_TIME_FPGA_TOUCH_FLASH_US (10 * 1000 * 1000) // 10s #define CMD_GET_FLASH "awk \'$4== \"\\\"bitfile(spi)\\\"\" {print $1}\' /proc/mtd" #define CMD_FPGA_PICKED_STATUS "cat /sys/class/gpio/gpio7/value" #define FLASH_BUFFER_MEMORY_SIZE (128 * 1024) // 500 KB // clang-format on #define FLASH_DRIVE_NAME_SIZE 16 char flashDriveName[FLASH_DRIVE_NAME_SIZE] = {0}; int gpioDefined = 0; extern int executeCommand(char *command, char *result, enum TLogLevel level); void defineGPIOpins() { #ifdef VIRTUAL return; #endif if (!gpioDefined) { // define the gpio pins system("echo 7 > /sys/class/gpio/export"); system("echo 9 > /sys/class/gpio/export"); // define their direction system("echo in > /sys/class/gpio/gpio7/direction"); system("echo out > /sys/class/gpio/gpio9/direction"); LOG(logINFO, ("gpio pins defined\n")); gpioDefined = 1; } else LOG(logDEBUG1, ("gpio pins already defined earlier\n")); } void FPGAdontTouchFlash() { #ifdef VIRTUAL return; #endif // tell FPGA to not touch flash system("echo 0 > /sys/class/gpio/gpio9/value"); // usleep(100*1000); } void FPGATouchFlash() { #ifdef VIRTUAL return; #endif // tell FPGA to touch flash to program itself system("echo 1 > /sys/class/gpio/gpio9/value"); } void resetFPGA() { LOG(logINFOBLUE, ("Reseting FPGA\n")); #ifdef VIRTUAL return; #endif FPGAdontTouchFlash(); FPGATouchFlash(); usleep(CTRL_SRVR_INIT_TIME_US); } int deleteOldFile(char *mess) { char cmd[MAX_STR_LENGTH] = {0}; char retvals[MAX_STR_LENGTH] = {0}; sprintf(cmd, "rm -fr %s", TEMP_PROG_FILE_NAME); if (FAIL == executeCommand(cmd, retvals, logDEBUG1)) { strcpy(mess, "Could not program fpga. (could not delete old file: "); strncat(mess, retvals, sizeof(mess) - strlen(mess) - 1); strcat(mess, "\n"); LOG(logERROR, (mess)); return FAIL; } LOG(logINFO, ("\tDeleted old programming file (%s)\n", TEMP_PROG_FILE_NAME)); return OK; } int preparetoCopyFPGAProgram(FILE **fd, uint64_t fsize, char *mess) { if (deleteOldFile(mess) == FAIL) { return FAIL; } // check available memory to copy program { struct sysinfo info; sysinfo(&info); if (fsize >= info.freeram) { sprintf(mess, "Could not program fpga. Not enough memory to copy " "program. [File size:%ldMB, free RAM: %ldMB]\n", (long int)(fsize / (1024 * 1024)), (long int)(info.freeram / (1024 * 1024))); LOG(logERROR, (mess)); return FAIL; } } // open file to copy program *fd = fopen(TEMP_PROG_FILE_NAME, "w"); if (*fd == NULL) { sprintf(mess, "Unable to open %s in write mode\n", TEMP_PROG_FILE_NAME); LOG(logERROR, (mess)); return FAIL; } LOG(logINFO, ("\tGoing to copy program to %s\n", TEMP_PROG_FILE_NAME)); return OK; } int copyToFlash(ssize_t fsize, char *clientChecksum, char *mess) { if (getDrive(mess) == FAIL) { return FAIL; } FILE *flashfd = NULL; FILE *srcfd = NULL; if (openFileForFlash(&flashfd, &srcfd, mess) == FAIL) { return FAIL; } if (eraseFlash(mess) == FAIL) { fclose(flashfd); fclose(srcfd); return FAIL; } if (writeToFlash(fsize, flashfd, srcfd, mess) == FAIL) { return FAIL; } if (deleteOldFile(mess) == FAIL) { return FAIL; } /* ignoring this until a consistent way to read from bfin flash if (verifyChecksumFromFlash(mess, clientChecksum, flashDriveName, fsize) == FAIL) { return FAIL; } */ if (waitForFPGAtoTouchFlash(mess) == FAIL) { return FAIL; } return OK; } int getDrive(char *mess) { #ifdef VIRTUAL strcpy(flashDriveName, "/tmp/SLS_mtd3"); return OK; #endif LOG(logDEBUG1, ("Finding flash drive...\n")); // getting the drive // root:/> cat /proc/mtd // dev: size erasesize name // mtd0: 00040000 00020000 "bootloader(nor)" // mtd1: 00100000 00020000 "linux kernel(nor)" // mtd2: 002c0000 00020000 "file system(nor)" // mtd3: 01000000 00010000 "bitfile(spi)" char cmd[MAX_STR_LENGTH] = {0}; char retvals[MAX_STR_LENGTH] = {0}; strcpy(cmd, CMD_GET_FLASH); if (executeCommand(cmd, retvals, logDEBUG1) == FAIL) { strcpy(mess, "Could not program fpga. (could not get flash drive: "); strncat(mess, retvals, sizeof(mess) - strlen(mess) - 1); strcat(mess, "\n"); LOG(logERROR, (mess)); return FAIL; } char *pch = strtok(retvals, ":"); if (pch == NULL) { strcpy(mess, "Could not get mtd drive to flash (strtok fail).\n"); LOG(logERROR, (mess)); return FAIL; } memset(flashDriveName, 0, sizeof(flashDriveName)); strcpy(flashDriveName, "/dev/"); strcat(flashDriveName, pch); LOG(logINFO, ("\tFlash drive found: %s\n", flashDriveName)); return OK; } int openFileForFlash(FILE **flashfd, FILE **srcfd, char *mess) { FPGAdontTouchFlash(); // open src file *srcfd = fopen(TEMP_PROG_FILE_NAME, "r"); if (*srcfd == NULL) { sprintf(mess, "Could not flash. Unable to open temp program file %s in read " "mode\n", TEMP_PROG_FILE_NAME); LOG(logERROR, (mess)); return FAIL; } LOG(logDEBUG1, ("Temp file ready for reading\n")); // open flash drive for writing *flashfd = fopen(flashDriveName, "w"); if (*flashfd == NULL) { fclose(*srcfd); sprintf(mess, "Unable to open flash drive %s in write mode\n", flashDriveName); LOG(logERROR, (mess)); return FAIL; } LOG(logINFO, ("\tFlash ready for writing\n")); return OK; } int eraseFlash(char *mess) { LOG(logINFO, ("\tErasing Flash...\n")); #ifdef VIRTUAL return OK; #endif char cmd[MAX_STR_LENGTH] = {0}; char retvals[MAX_STR_LENGTH] = {0}; sprintf(cmd, "flash_eraseall %s", flashDriveName); if (FAIL == executeCommand(cmd, retvals, logDEBUG1)) { strcpy(mess, "Could not program fpga. (could not erase flash: "); strncat(mess, retvals, sizeof(mess) - strlen(mess) - 1); strcat(mess, "\n"); LOG(logERROR, (mess)); return FAIL; } LOG(logINFO, ("\tFlash erased\n")); return OK; } int writeToFlash(ssize_t fsize, FILE *flashfd, FILE *srcfd, char *mess) { LOG(logDEBUG1, ("writing to flash\n")); char* buffer = malloc(FLASH_BUFFER_MEMORY_SIZE); if (buffer == NULL) { fclose(flashfd); fclose(srcfd); strcpy(mess, "Could not program fpga. Memory allocation to write to " "flash failed.\n"); LOG(logERROR, (mess)); return FAIL; } LOG(logINFO, ("\tWriting to Flash...\n")); int oldProgress = 0; ssize_t totalBytes = 0; ssize_t bytes = fread((void*)buffer, sizeof(char), FLASH_BUFFER_MEMORY_SIZE, srcfd); while (bytes > 0) { ssize_t bytesWritten = fwrite((void*)buffer, sizeof(char), bytes, flashfd); totalBytes += bytesWritten; if (bytesWritten != bytes) { free(buffer); fclose(flashfd); fclose(srcfd); sprintf(mess, "Could not write to flash (bytes written:%ld, expected: " "%ld, total written:%ld)\n", (long int)bytesWritten, (long int)bytes, (long int)totalBytes); LOG(logERROR, (mess)); return FAIL; } // print progress if (fsize > 0) { int progress = (int)(((double)(totalBytes) / fsize) * 100); if (oldProgress != progress) { printf("%d%%\r", progress); fflush(stdout); oldProgress = progress; } } else printf("."); bytes = fread((void*)buffer, sizeof(char), FLASH_BUFFER_MEMORY_SIZE, srcfd); } if (fsize <= 0) { printf("\n"); } free(buffer); fclose(flashfd); fclose(srcfd); LOG(logINFO, ("\tWrote %ld bytes to flash\n", totalBytes)); if (totalBytes != fsize) { sprintf(mess, "Could not program fpga. Incorrect bytes written to flash %lu [expected: %lu]\n", totalBytes, fsize); LOG(logERROR, (mess)); return FAIL; } return OK; } int waitForFPGAtoTouchFlash(char* mess) { // touch and program FPGATouchFlash(); #ifdef VIRTUAL return OK; #endif LOG(logINFO, ("\tWaiting for FPGA to program from flash\n")); int timeSpent = 0; int result = 0; while (result == 0) { // time taken for fpga to pick up from flash usleep(1000); timeSpent += 1000; if (timeSpent >= MAX_TIME_FPGA_TOUCH_FLASH_US) { sprintf(mess, "Could not program fpga. (exceeded max time allowed: %ds)\n", MAX_TIME_FPGA_TOUCH_FLASH_US/(1000 * 1000)); LOG(logERROR, (mess)); return FAIL; } // read gpio status char retvals[MAX_STR_LENGTH] = {0}; if (FAIL == executeCommand(CMD_FPGA_PICKED_STATUS, retvals, logDEBUG1)) { strcpy(mess, "Could not program fpga. (could not read gpio status: "); strncat(mess, retvals, sizeof(mess) - strlen(mess) - 1); strcat(mess, "\n"); LOG(logERROR, (mess)); return FAIL; } // convert to int if (sscanf(retvals, "%d\n", &result) != 1) { sprintf(mess, "Could not program fpga. (could not scan int for gpio status: [%s])\n", retvals); LOG(logERROR, (mess)); return FAIL; } LOG(logDEBUG1, ("gpi07 returned %d\n", result)); } LOG(logINFO, ("\tFPGA has picked up the program from flash\n")); return OK; }