#include "programFpgaBlackfin.h" #include "clogger.h" #include "common.h" #include "sls/ansi.h" #include "slsDetectorServer_defs.h" #include #include // usleep #include /* global variables */ #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_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 preparetoCopyFPGAProgram(FILE **fd, uint64_t fsize, char *mess) { // delete old /var/tmp/file { 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; } } // 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, ("\tCopying program to %s\n", TEMP_PROG_FILE_NAME)); return OK; } int copyToFlash(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) { return FAIL; } if (writeToFlash(flashfd, srcfd, mess) == FAIL) { return FAIL; } if (verifyChecksumFromFile(mess, clientChecksum, flashDriveName) == FAIL) { return FAIL; } LOG(logINFO, ("Checksum in Flash verified\n")); if (waitForFPGAtoTouchFlash(mess) == FAIL) { return FAIL; } return OK; } int getDrive(char *mess) { #ifdef VIRTUAL strcpy(flashDriveName, "/tmp/SLS_mtd3"); return OK; #endif // 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 retvals[MAX_STR_LENGTH] = {0}; if (executeCommand(CMD_GET_FLASH, 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; } 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(logDEBUG1, ("Erasing 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(FILE *flashfd, FILE *srcfd, char *mess) { LOG(logDEBUG1, ("writing to flash\n")); char* buffer = malloc(MAX_FPGAPROGRAMSIZE); if (buffer == NULL) { fclose(flashfd); fclose(srcfd); strcpy(mess, "Could not program foga. Memory allocation to write to " "flash failed.\n"); LOG(logERROR, (mess)); return FAIL; } ssize_t totalBytes = 0; ssize_t bytes = fread(buffer, sizeof(char), MAX_FPGAPROGRAMSIZE, srcfd); while (bytes > 0) { ssize_t bytesWritten = fwrite((void *)buffer, sizeof(char), bytes, flashfd); totalBytes += bytesWritten; if (bytesWritten != bytes) { 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; } bytes = fread(buffer, sizeof(char), bytes, srcfd); printf("."); } printf("\n"); fclose(flashfd); fclose(srcfd); LOG(logINFO, ("\tWrote %ld bytes to flash\n", totalBytes)); return OK; } int waitForFPGAtoTouchFlash(char* mess) { // touch and program FPGATouchFlash(); #ifdef VIRTUAL return OK; #endif LOG(logINFO, ("Waiting 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", &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; }