/* Copyright 2019, The Regents of the University of California. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE REGENTS OF THE UNIVERSITY OF CALIFORNIA ''AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OF THE UNIVERSITY OF CALIFORNIA OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. The views and conclusions contained in the software and documentation are those of the authors and should not be interpreted as representing official policies, either expressed or implied, of The Regents of the University of California. */ #include #include #include #include #include #include #include #include #include #include #include "mqnic.h" #include "bitfile.h" #include "flash.h" #include "fpga_id.h" #define MAX_SEGMENTS 8 uint32_t reverse_bits_32(uint32_t x) { x = ((x & 0x55555555) << 1) | ((x & 0xAAAAAAAA) >> 1); x = ((x & 0x33333333) << 2) | ((x & 0xCCCCCCCC) >> 2); x = ((x & 0x0F0F0F0F) << 4) | ((x & 0xF0F0F0F0) >> 4); x = ((x & 0x00FF00FF) << 8) | ((x & 0xFF00FF00) >> 8); x = ((x & 0x0000FFFF) << 16) | ((x & 0xFFFF0000) >> 16); return x; } uint16_t reverse_bits_16(uint16_t x) { x = ((x & 0x5555) << 1) | ((x & 0xAAAA) >> 1); x = ((x & 0x3333) << 2) | ((x & 0xCCCC) >> 2); x = ((x & 0x0F0F) << 4) | ((x & 0xF0F0) >> 4); x = ((x & 0x00FF) << 8) | ((x & 0xFF00) >> 8); return x; } uint8_t reverse_bits_8(uint8_t x) { x = ((x & 0x55) << 1) | ((x & 0xAA) >> 1); x = ((x & 0x33) << 2) | ((x & 0xCC) >> 2); x = ((x & 0x0F) << 4) | ((x & 0xF0) >> 4); return x; } char* stristr(const char *str1, const char *str2) { const char* p1 = str1; const char* p2 = str2; const char* r = *p2 == 0 ? str1 : 0; while (*p1 != 0 && *p2 != 0) { if (tolower(*p1) == tolower(*p2)) { if (r == 0) { r = p1; } p2++; } else { p2 = str2; if (r != 0) { p1 = r + 1; } if (tolower(*p1) == tolower(*p2)) { r = p1; p2++; } else { r = 0; } } p1++; } return *p2 == 0 ? (char *)r : 0; } static void usage(char *name) { fprintf(stderr, "usage: %s [options]\n" " -d name device to open (/dev/mqnic0)\n" " -s slot slot to program (default 1)\n" " -r file read flash to file\n" " -w file write and verify flash from file\n" " -b boot FPGA from flash\n" " -t hot reset FPGA\n", name); } int flash_read_progress(struct flash_device *fdev, size_t addr, size_t len, void* dest) { int ret; size_t remain = len; size_t seg; int step = 0x10000; printf("Start address: 0x%08lx\n", addr); printf("Length: 0x%08lx\n", len); while (remain > 0) { if (remain > step) { // longer than step, trim if ((addr + step) & (step-1)) { // align to step size seg = step - ((addr + step) & (step-1)); } else { // already aligned seg = step; } } else { // shorter than step seg = remain; } printf("Read address 0x%08lx, length 0x%08lx (%ld%%)\r", addr, seg, ((100*(len-remain))/len)); fflush(stdout); ret = flash_read(fdev, addr, seg, dest); if (ret) return ret; addr += seg; remain -= seg; dest += seg; } printf("\n"); return 0; } int flash_write_progress(struct flash_device *fdev, size_t addr, size_t len, void* src) { int ret; size_t remain = len; size_t seg; int step = 0x10000; printf("Start address: 0x%08lx\n", addr); printf("Length: 0x%08lx\n", len); step = fdev->write_buffer_size > step ? fdev->write_buffer_size : step; while (remain > 0) { if (remain > step) { // longer than step, trim if ((addr + step) & (step-1)) { // align to step size seg = step - ((addr + step) & (step-1)); } else { // already aligned seg = step; } } else { // shorter than step seg = remain; } printf("Write address 0x%08lx, length 0x%08lx (%ld%%)\r", addr, seg, ((100*(len-remain))/len)); fflush(stdout); ret = flash_write(fdev, addr, seg, src); if (ret) return ret; addr += seg; remain -= seg; src += seg; } printf("\n"); return 0; } int flash_erase_progress(struct flash_device *fdev, size_t addr, size_t len) { int ret; size_t remain = len; size_t seg; int step = 0x10000; printf("Start address: 0x%08lx\n", addr); printf("Length: 0x%08lx\n", len); step = fdev->erase_block_size > step ? fdev->erase_block_size : step; while (remain > 0) { if (remain > step) { // longer than step, trim if ((addr + step) & (step-1)) { // align to step size seg = step - ((addr + step) & (step-1)); } else { // already aligned seg = step; } } else { // shorter than step seg = remain; } printf("Erase address 0x%08lx, length 0x%08lx (%ld%%)\r", addr, seg, ((100*(len-remain))/len)); fflush(stdout); ret = flash_erase(fdev, addr, seg); if (ret) return ret; addr += seg; remain -= seg; } printf("\n"); return 0; } int write_str_to_file(const char *file_name, const char *str) { int ret = 0; FILE *fp = fopen(file_name, "w"); if (!fp) { perror("failed to open file"); return -1; } if (fputs(str, fp) == EOF) { perror("failed to write to file"); ret = -1; } fclose(fp); return ret; } int write_1_to_file(const char *file_name) { return write_str_to_file(file_name, "1"); } #define FILE_TYPE_BIN 0 #define FILE_TYPE_HEX 1 #define FILE_TYPE_BIT 2 int file_type_from_ext(const char *file_name) { char *ptr; char buffer[32]; ptr = strrchr(file_name, '.'); if (!ptr) { return FILE_TYPE_BIN; } ptr++; for (int i = 0; i < sizeof(buffer)-1 && *ptr; i++) { buffer[i] = tolower(*ptr++); buffer[i+1] = 0; } if (strcmp(buffer, "hex") == 0 || strcmp(buffer, "mcs") == 0) { return FILE_TYPE_HEX; } if (strcmp(buffer, "bit") == 0) { return FILE_TYPE_BIT; } return FILE_TYPE_BIN; } int pcie_hot_reset(const char *pcie_port_path) { int fd; char path[PATH_MAX+32]; char buf[32]; snprintf(path, sizeof(path), "%s/config", pcie_port_path); fd = open(path, O_RDWR); if (!fd) { perror("Failed to open config region of port"); return -1; } // set and then clear secondary bus reset bit (mask 0x0040) // in the bridge control register (offset 0x3e) pread(fd, buf, 2, PCI_BRIDGE_CONTROL); buf[2] = buf[0] | PCI_BRIDGE_CTL_BUS_RESET; buf[3] = buf[1]; pwrite(fd, buf+2, 2, PCI_BRIDGE_CONTROL); usleep(10000); pwrite(fd, buf, 2, PCI_BRIDGE_CONTROL); close(fd); return 0; } int pcie_disable_fatal_err(const char *pcie_port_path) { int fd; char path[PATH_MAX+32]; char buf[32]; int offset; snprintf(path, sizeof(path), "%s/config", pcie_port_path); fd = open(path, O_RDWR); if (!fd) { perror("Failed to open config region of port"); return -1; } // clear SERR bit (mask 0x0100) in command register (offset 0x04) pread(fd, buf, 2, PCI_COMMAND); buf[1] &= ~(PCI_COMMAND_SERR >> 8); pwrite(fd, buf, 2, PCI_COMMAND); // clear fatal error reporting bit (mask 0x0004) in // PCIe capability device control register (offset 0x08) // find PCIe capability (ID 0x10) pread(fd, buf, 1, PCI_CAPABILITY_LIST); offset = buf[0] & 0xfc; while (offset > 0) { pread(fd, buf, 2, offset); if (buf[0] == PCI_CAP_ID_EXP) break; offset = buf[1] & 0xfc; } // clear bit if (offset) { pread(fd, buf, 2, offset+PCI_EXP_DEVCTL); buf[0] &= ~PCI_EXP_DEVCTL_FERE; pwrite(fd, buf, 2, offset+PCI_EXP_DEVCTL); } close(fd); return 0; } int main(int argc, char *argv[]) { char *name; int opt; int ret = 0; char *device = NULL; char *read_file_name = NULL; FILE *read_file = NULL; char *write_file_name = NULL; FILE *write_file = NULL; char path[PATH_MAX+32]; char device_path[PATH_MAX]; char port_path[PATH_MAX]; char *ptr; int slot = -1; char action_read = 0; char action_write = 0; char action_boot = 0; char action_reset = 0; struct mqnic *dev = NULL; struct flash_device *pri_flash = NULL; struct flash_device *sec_flash = NULL; int flash_segment_count = 0; size_t flash_segment_start[MAX_SEGMENTS]; size_t flash_segment_length[MAX_SEGMENTS]; name = strrchr(argv[0], '/'); name = name ? 1+name : argv[0]; while ((opt = getopt(argc, argv, "d:s:r:w:bth?")) != EOF) { switch (opt) { case 'd': device = optarg; break; case 's': slot = atoi(optarg); break; case 'r': action_read = 1; read_file_name = optarg; break; case 'w': action_write = 1; write_file_name = optarg; break; case 'b': action_boot = 1; action_reset = 1; break; case 't': action_reset = 1; break; case 'h': case '?': usage(name); return 0; default: usage(name); return -1; } } if (!device) { fprintf(stderr, "Device not specified\n"); usage(name); return -1; } dev = mqnic_open(device); if (!dev) { fprintf(stderr, "Failed to open device\n"); return -1; } // determine sysfs path of PCIe device // first, try to find via miscdevice ptr = strrchr(device, '/'); ptr = ptr ? ptr+1 : device; snprintf(path, sizeof(path), "/sys/class/misc/%s/device", ptr); if (!realpath(path, device_path)) { // that failed, perhaps it was a PCIe resource strcpy(path, device); ptr = strrchr(path, '/'); if (ptr) *ptr = 0; if (!realpath(path, device_path)) { perror("failed to determine device path"); ret = -1; goto err; } } // PCIe device will have a config space, so check for that snprintf(path, sizeof(path), "%s/config", device_path); if (access(path, F_OK) == -1) { perror("failed to determine device path"); ret = -1; goto err; } // determine sysfs path of upstream port strcpy(port_path, device_path); ptr = strrchr(port_path, '/'); if (ptr) *ptr = 0; printf("PCIe ID (device): %s\n", strrchr(device_path, '/')+1); printf("PCIe ID (upstream port): %s\n", strrchr(port_path, '/')+1); printf("FW ID: 0x%08x\n", dev->fw_id); printf("FW version: %d.%d\n", dev->fw_ver >> 16, dev->fw_ver & 0xffff); printf("Board ID: 0x%08x\n", dev->board_id); printf("Board version: %d.%d\n", dev->board_ver >> 16, dev->board_ver & 0xffff); uint32_t flash_id = mqnic_reg_read32(dev->regs, MQNIC_REG_FLASH_ID); uint32_t fpga_id = mqnic_reg_read32(dev->regs, MQNIC_REG_FPGA_ID); const char *fpga_part = get_fpga_part(fpga_id); uint8_t flash_type = flash_id >> 0; uint8_t flash_configuration = flash_id >> 8; uint8_t flash_data_width = flash_id >> 16; uint8_t flash_addr_width = flash_id >> 24; printf("Flash ID: 0x%08x\n", flash_id); printf("FPGA ID: 0x%08x\n", fpga_id); printf("FPGA part: %s\n", fpga_part); if (flash_id == 0 || flash_id == 0xffffffff) { fprintf(stderr, "Invalid flash ID\n"); ret = -1; goto skip_flash; } if (fpga_id == 0 || fpga_id == 0xffffffff) { fprintf(stderr, "Invalid FPGA ID\n"); ret = -1; goto skip_flash; } int bitswap = 0; int word_size = 8; int dual_qspi = 0; size_t flash_size; size_t segment_size; size_t segment_offset; if (flash_type == 0 || flash_type == 2) { printf("Flash type: SPI\n"); printf("Data width: %d\n", flash_data_width); if (flash_data_width > 4) { dual_qspi = 1; pri_flash = flash_open_spi(4, dev->regs+MQNIC_REG_FLASH_SPI_0_CTRL); sec_flash = flash_open_spi(4, dev->regs+MQNIC_REG_FLASH_SPI_1_CTRL); if (!pri_flash || !sec_flash) { fprintf(stderr, "Failed to connect to flash device\n"); ret = -1; goto skip_flash; } flash_size = pri_flash->size+sec_flash->size; } else { pri_flash = flash_open_spi(flash_data_width, dev->regs+MQNIC_REG_FLASH_SPI_0_CTRL); if (!pri_flash) { fprintf(stderr, "Failed to connect to flash device\n"); ret = -1; goto skip_flash; } flash_size = pri_flash->size; } } else if (flash_type == 1) { printf("Flash type: BPI\n"); printf("Data width: %d\n", flash_data_width); printf("Address width: %d\n", flash_addr_width); bitswap = 1; if (flash_data_width == 16) { word_size = 16; } pri_flash = flash_open_bpi(flash_data_width, dev->regs+MQNIC_REG_FLASH_BPI_CTRL, dev->regs+MQNIC_REG_FLASH_BPI_ADDR, dev->regs+MQNIC_REG_FLASH_BPI_DATA); if (!pri_flash) { fprintf(stderr, "Failed to connect to flash device\n"); ret = -1; goto skip_flash; } flash_size = pri_flash->size; } else { fprintf(stderr, "Unknown flash type: %d\n", flash_type); ret = -1; goto skip_flash; } switch (flash_configuration) { case 0: case 1: flash_segment_count = 1; flash_segment_start[0] = 0; flash_segment_length[0] = flash_size; break; case 2: flash_segment_count = 2; flash_segment_start[0] = 0; flash_segment_length[0] = flash_size >> 1; flash_segment_start[1] = flash_segment_start[0]+flash_segment_length[0]; flash_segment_length[1] = flash_size >> 1; break; case 4: flash_segment_count = 4; flash_segment_start[0] = 0; flash_segment_length[0] = flash_size >> 2; for (int k = 1; k < 4; k++) { flash_segment_start[k] = flash_segment_start[k-1]+flash_segment_length[k-1]; flash_segment_length[k] = flash_size >> 2; } break; case 8: flash_segment_count = 8; flash_segment_start[0] = 0; flash_segment_length[0] = flash_size >> 3; for (int k = 1; k < 8; k++) { flash_segment_start[k] = flash_segment_start[k-1]+flash_segment_length[k-1]; flash_segment_length[k] = flash_size >> 3; } break; case 0x81: // Alveo boards if (flash_size < 0x01002000) { fprintf(stderr, "Invalid flash size\n"); ret = -1; goto skip_flash; } flash_segment_count = 2; flash_segment_start[0] = 0; flash_segment_length[0] = 0x01002000; flash_segment_start[1] = flash_segment_start[0]+flash_segment_length[0]; flash_segment_length[1] = flash_size - flash_segment_start[1]; break; default: fprintf(stderr, "Unknown flash configuration (0x%02x)\n", flash_configuration); ret = -1; goto skip_flash; } for (int k = 0; k < flash_segment_count; k++) { printf("Flash segment %d: start 0x%08lx length 0x%08lx\n", k, flash_segment_start[k], flash_segment_length[k]); } if (slot < 0) { if (flash_segment_count > 1) { slot = 1; } else { slot = 0; } } if ((action_read || action_write) && (slot < 0 || slot >= flash_segment_count)) { fprintf(stderr, "Requested slot is not valid (%d)\n", slot); ret = -1; goto err; } segment_offset = flash_segment_start[slot]; segment_size = flash_segment_length[slot]; printf("Selected: segment %d start 0x%08lx length 0x%08lx\n", slot, segment_offset, segment_size); if (action_write) { char *segment = calloc(segment_size, 1); memset(segment, 0xff, segment_size); size_t len; int file_type = file_type_from_ext(write_file_name); if (file_type == FILE_TYPE_BIN) { // read binary file printf("Reading binary file \"%s\"...\n", write_file_name); write_file = fopen(write_file_name, "rb"); if (!write_file) { fprintf(stderr, "Failed to open file\n"); free(segment); ret = -1; goto err; } fseek(write_file, 0, SEEK_END); len = ftell(write_file); rewind(write_file); if (len > segment_size) { fprintf(stderr, "File larger than segment (%ld > %ld)\n", len, segment_size); fclose(write_file); free(segment); ret = -1; goto err; } if (fread(segment, 1, len, write_file) < len) { fprintf(stderr, "Error reading file\n"); fclose(write_file); free(segment); ret = -1; goto err; } fclose(write_file); } else if (file_type == FILE_TYPE_BIT) { // read bit file struct bitfile *bf; bf = bitfile_create_from_file(write_file_name); if (!bf) { fprintf(stderr, "Error reading bit file\n"); free(segment); ret = -1; goto err; } if (stristr(bf->part, fpga_part) != bf->part) { fprintf(stderr, "Device mismatch (target is %s, file is %s)\n", fpga_part, bf->part); bitfile_close(bf); free(segment); ret = -1; goto err; } if (bf->data_len > segment_size) { fprintf(stderr, "File larger than segment (%ld > %ld)\n", bf->data_len, segment_size); bitfile_close(bf); free(segment); ret = -1; goto err; } len = bf->data_len; memcpy(segment, bf->data, bf->data_len); bitfile_close(bf); } else if (file_type == FILE_TYPE_HEX) { fprintf(stderr, "Hex files are not currently supported\n"); free(segment); ret = -1; goto err; } else { fprintf(stderr, "Unsupported file type\n"); free(segment); ret = -1; goto err; } // check sync word if (memcmp(segment+0x50, "\xAA\x99\x55\x66", 4)) { fprintf(stderr, "Bitstream sync word not found\n"); free(segment); ret = -1; goto err; } // TODO check for and confirm FPGA ID if (bitswap) { if (word_size == 16) { uint16_t *p = (uint16_t *)segment; for (size_t k = 0; k < segment_size; k += 2) { *p = reverse_bits_16(*p); p++; } } else { uint8_t *p = (uint8_t *)segment; for (size_t k = 0; k < segment_size; k++) { *p = reverse_bits_8(*p); p++; } } } if (dual_qspi) { // Dual QSPI flash // check sync word for dual QSPI re-sync if (memcmp(segment+0x70, "\xAA\x99\x55\x66", 4)) { fprintf(stderr, "Bitstream sync word not found for dual QSPI re-sync\n"); free(segment); ret = -1; goto err; } char *pri_buf = calloc(segment_size/2, 1); char *sec_buf = calloc(segment_size/2, 1); memset(pri_buf, 0xff, segment_size/2); memset(sec_buf, 0xff, segment_size/2); int offset = 0x68; size_t len_int = (len - offset) / 2 + offset; if (len_int > segment_size/2) len_int = segment_size/2; memcpy(pri_buf, segment, offset); char *c1 = pri_buf+offset; char *c2 = sec_buf+offset; for (size_t k = offset; k < segment_size-offset; k += 2) { *c1 = (segment[k+1] & 0x0f) | ((segment[k] << 4) & 0xf0); *c2 = ((segment[k+1] >> 4) & 0x0f) | (segment[k] & 0xf0); c1++; c2++; } // round up length to block size if ((segment_offset/2 + len_int) & (pri_flash->erase_block_size-1)) { len_int += pri_flash->erase_block_size - ((segment_offset/2 + len_int) & (pri_flash->erase_block_size-1)); } printf("Erasing primary flash...\n"); if (flash_erase_progress(pri_flash, segment_offset/2, len_int)) { fprintf(stderr, "Erase failed!\n"); ret = -1; free(segment); free(pri_buf); free(sec_buf); goto err; } printf("Erasing secondary flash...\n"); if (flash_erase_progress(sec_flash, segment_offset/2, len_int)) { fprintf(stderr, "Erase failed!\n"); ret = -1; free(segment); free(pri_buf); free(sec_buf); goto err; } printf("Writing primary flash...\n"); if (flash_write_progress(pri_flash, segment_offset/2, len_int, pri_buf)) { fprintf(stderr, "Write failed!\n"); ret = -1; free(segment); free(pri_buf); free(sec_buf); goto err; } printf("Writing secondary flash...\n"); if (flash_write_progress(sec_flash, segment_offset/2, len_int, sec_buf)) { fprintf(stderr, "Write failed!\n"); ret = -1; free(segment); free(pri_buf); free(sec_buf); goto err; } char *pri_check_buf = calloc(segment_size/2, 1); char *sec_check_buf = calloc(segment_size/2, 1); memset(pri_check_buf, 0xff, segment_size/2); memset(sec_check_buf, 0xff, segment_size/2); printf("Verifying primary flash...\n"); flash_read_progress(pri_flash, segment_offset/2, len_int, pri_check_buf); printf("Verifying secondary flash...\n"); flash_read_progress(sec_flash, segment_offset/2, len_int, sec_check_buf); if (memcmp(pri_buf, pri_check_buf, len_int) || memcmp(sec_buf, sec_check_buf, len_int)) { fprintf(stderr, "Verify failed!\n"); for (size_t k = 0; k < len; k++) { if (pri_buf[k] != pri_check_buf[k]) { fprintf(stderr, "primary flash offset 0x%08lx: expected 0x%02x, read 0x%02x\n", k, pri_buf[k] & 0xff, pri_check_buf[k] & 0xff); } } for (size_t k = 0; k < len; k++) { if (sec_buf[k] != sec_check_buf[k]) { fprintf(stderr, "secondary flash offset 0x%08lx: expected 0x%02x, read 0x%02x\n", k, sec_buf[k] & 0xff, sec_check_buf[k] & 0xff); } } ret = -1; } else { printf("Programming succeeded!\n"); } free(pri_check_buf); free(sec_check_buf); free(pri_buf); free(sec_buf); } else { // SPI or BPI flash // round up length to block size if ((segment_offset + len) & (pri_flash->erase_block_size-1)) { len += pri_flash->erase_block_size - ((segment_offset + len) & (pri_flash->erase_block_size-1)); } printf("Erasing flash...\n"); if (flash_erase_progress(pri_flash, segment_offset, len)) { fprintf(stderr, "Erase failed!\n"); ret = -1; free(segment); goto err; } printf("Writing flash...\n"); if (flash_write_progress(pri_flash, segment_offset, len, segment)) { fprintf(stderr, "Write failed!\n"); ret = -1; free(segment); goto err; } char *check_buf = calloc(segment_size, 1); memset(check_buf, 0xff, segment_size); printf("Verifying flash...\n"); flash_read_progress(pri_flash, segment_offset, len, check_buf); if (memcmp(segment, check_buf, len)) { fprintf(stderr, "Verify failed!\n"); for (size_t k = 0; k < len; k++) { if (segment[k] != check_buf[k]) { fprintf(stderr, "flash offset 0x%08lx: expected 0x%02x, read 0x%02x\n", k, segment[k] & 0xff, check_buf[k] & 0xff); } } ret = -1; } else { printf("Programming succeeded!\n"); } free(check_buf); } free(segment); } if (action_read) { char *segment = calloc(segment_size, 1); memset(segment, 0xff, segment_size); if (dual_qspi) { char *pri_buf = calloc(segment_size/2, 1); char *sec_buf = calloc(segment_size/2, 1); printf("Reading primary flash...\n"); flash_read_progress(pri_flash, segment_offset/2, segment_size/2, pri_buf); printf("Reading secondary flash...\n"); flash_read_progress(sec_flash, segment_offset/2, segment_size/2, sec_buf); int offset = 0x68; memcpy(segment, pri_buf, offset); char *c1 = pri_buf+offset; char *c2 = sec_buf+offset; for (size_t k = offset; k < segment_size-offset; k += 2) { segment[k] = ((*c1 >> 4) & 0x0f) | (*c2 & 0xf0); segment[k+1] = (*c1 & 0x0f) | ((*c2 << 4) & 0xf0); c1++; c2++; } free(pri_buf); free(sec_buf); } else { printf("Reading flash...\n"); flash_read_progress(pri_flash, segment_offset, segment_size, segment); } if (bitswap) { if (word_size == 16) { uint16_t *p = (uint16_t *)segment; for (size_t k = 0; k < segment_size; k += 2) { *p = reverse_bits_16(*p); p++; } } else { uint8_t *p = (uint8_t *)segment; for (size_t k = 0; k < segment_size; k++) { *p = reverse_bits_8(*p); p++; } } } int file_type = file_type_from_ext(read_file_name); if (file_type == FILE_TYPE_BIN) { // write binary file printf("Writing binary file \"%s\"...\n", read_file_name); read_file = fopen(read_file_name, "wb"); fwrite(segment, 1, segment_size, read_file); fclose(read_file); } else if (file_type == FILE_TYPE_HEX) { fprintf(stderr, "Hex files are not currently supported\n"); free(segment); ret = -1; goto err; } else { fprintf(stderr, "Unsupported file type\n"); free(segment); ret = -1; goto err; } free(segment); } skip_flash: if (ret && (action_read || action_write)) { goto err; } else { ret = 0; } flash_release(pri_flash); pri_flash = NULL; flash_release(sec_flash); sec_flash = NULL; if (action_boot || action_reset) { printf("Preparing to reset device...\n"); // disable fatal error reporting on port (to prevent IPMI-triggered reboot) printf("Disabling PCIe fatal error reporting on port...\n"); pcie_disable_fatal_err(port_path); // disconnect from device mqnic_close(dev); dev = NULL; // attempt to disconnect driver ptr = strrchr(device_path, '/'); if (ptr) { snprintf(path, sizeof(path), "%s/driver/unbind", device_path); if (access(path, F_OK) != -1) { printf("Unbinding driver...\n"); write_str_to_file(path, ptr+1); } else { printf("No driver bound\n"); } } sleep(1); // trigger FPGA reload if (action_boot) { // reconnect directly to device snprintf(path, sizeof(path), "%s/resource0", device_path); dev = mqnic_open(path); if (!dev) { fprintf(stderr, "Failed to open device\n"); ret = -1; goto err; } // reload FPGA printf("Triggering IPROG to reload FPGA...\n"); mqnic_reg_write32(dev->regs, MQNIC_REG_FPGA_ID, 0xFEE1DEAD); // disconnect mqnic_close(dev); dev = NULL; } // remove PCIe device printf("Removing device...\n"); snprintf(path, sizeof(path), "%s/remove", device_path); if (write_1_to_file(path)) { fprintf(stderr, "Failed to remove device!\n"); ret = -1; goto err; } if (action_boot) { // give FPGA some time to boot from flash sleep(4); } sleep(1); for (int tries = 5; tries > 0; tries--) { printf("Performing hot reset on upstream port...\n"); pcie_hot_reset(port_path); sleep(2); printf("Rescanning on upstream port...\n"); snprintf(path, sizeof(path), "%s/rescan", port_path); if (write_1_to_file(path)) { fprintf(stderr, "Rescan failed!\n"); ret = -1; goto err; } // PCIe device will have a config space, so check for that snprintf(path, sizeof(path), "%s/config", device_path); if (access(path, F_OK) == 0) { printf("Success, device is online!\n"); break; } else { if (tries > 0) { printf("Rescan failed, attempting another reset (up to %d more)\n", tries); } else { fprintf(stderr, "Rescan failed, device is offline!\n"); ret = -1; goto err; } } } } err: flash_release(pri_flash); flash_release(sec_flash); mqnic_close(dev); return ret; }