From e941d4d3f1796f7bca429c5ea8c7e2337304c21a Mon Sep 17 00:00:00 2001 From: Dawid Buchwald Date: Sun, 12 Dec 2021 20:00:23 +0000 Subject: [PATCH] Implemented faster flash programming method git-svn-id: svn://svn.savannah.nongnu.org/avrdude/branches/serialupdi@1517 81a1dc3b-b13d-400b-aceb-764788c761c2 --- avrdude/serialupdi.c | 2 + avrdude/updi_link.c | 103 +++++++++++++++++++++++++++++++++++++++ avrdude/updi_link.h | 1 + avrdude/updi_readwrite.c | 6 +-- 4 files changed, 107 insertions(+), 5 deletions(-) diff --git a/avrdude/serialupdi.c b/avrdude/serialupdi.c index 2db77875..98b09b32 100644 --- a/avrdude/serialupdi.c +++ b/avrdude/serialupdi.c @@ -489,6 +489,7 @@ static int serialupdi_paged_write(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m, rc = updi_nvm_write_flash(pgm, p, m->offset + write_offset, m->buf + write_offset, remaining_bytes > m->page_size ? m->page_size : remaining_bytes); } else { + avrdude_message(MSG_INFO, "%s: Invalid memory type: <%s:%d>, 0x%06X, %d (0x%04X)\n", progname, m->desc, page_size, addr, n_bytes, n_bytes); rc = -1; } @@ -508,6 +509,7 @@ static int serialupdi_paged_write(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m, } else if (strcmp(m->desc, "flash")==0) { rc = updi_nvm_write_flash(pgm, p, m->offset+addr, m->buf+addr, n_bytes); } else { + avrdude_message(MSG_INFO, "%s: Invalid memory type: <%s:%d>, 0x%06X, %d (0x%04X)\n", progname, m->desc, page_size, addr, n_bytes, n_bytes); rc = -1; } return rc; diff --git a/avrdude/updi_link.c b/avrdude/updi_link.c index 33c3da1d..a3852bfe 100644 --- a/avrdude/updi_link.c +++ b/avrdude/updi_link.c @@ -526,6 +526,109 @@ int updi_link_st_ptr_inc16(PROGRAMMER * pgm, unsigned char * buffer, uint16_t wo return 0; } +int updi_link_st_ptr_inc16_RSD(PROGRAMMER * pgm, unsigned char * buffer, uint16_t words, int blocksize) { +/* + def st_ptr_inc16_RSD(self, data, blocksize): + """ + Store a 16-bit word value to the pointer location with pointer post-increment + :param data: data to store + :blocksize: max number of bytes being sent -1 for all. + Warning: This does not strictly honor blocksize for values < 6 + We always glob together the STCS(RSD) and REP commands. + But this should pose no problems for compatibility, because your serial adapter can't deal with 6b chunks, + none of pymcuprog would work! + """ + self.logger.debug("ST16 to *ptr++ with RSD, data length: 0x%03X in blocks of: %d", len(data), blocksize) + + #for performance we glob everything together into one USB transfer.... + repnumber= ((len(data) >> 1) -1) + data = [*data, *[constants.UPDI_PHY_SYNC, constants.UPDI_STCS | constants.UPDI_CS_CTRLA, 0x06]] + + if blocksize == -1 : + # Send whole thing at once stcs + repeat + st + (data + stcs) + blocksize = 3 + 3 + 2 + len(data) + num = 0 + firstpacket = [] + if blocksize < 10 : + # very small block size - we send pair of 2-byte commands first. + firstpacket = [*[constants.UPDI_PHY_SYNC, constants.UPDI_STCS | constants.UPDI_CS_CTRLA, 0x0E], + *[constants.UPDI_PHY_SYNC, constants.UPDI_REPEAT | constants.UPDI_REPEAT_BYTE, (repnumber & 0xFF)]] + data = [*[constants.UPDI_PHY_SYNC, constants.UPDI_ST | constants.UPDI_PTR_INC |constants.UPDI_DATA_16], *data] + num = 0 + else: + firstpacket = [*[constants.UPDI_PHY_SYNC, constants.UPDI_STCS | constants.UPDI_CS_CTRLA , 0x0E], + *[constants.UPDI_PHY_SYNC, constants.UPDI_REPEAT | constants.UPDI_REPEAT_BYTE, (repnumber & 0xFF)], + *[constants.UPDI_PHY_SYNC, constants.UPDI_ST | constants.UPDI_PTR_INC | constants.UPDI_DATA_16], + *data[:blocksize - 8]] + num = blocksize - 8 + self.updi_phy.send( firstpacket ) + + # if finite block size, this is used. + while num < len(data): + data_slice = data[num:num+blocksize] + self.updi_phy.send(data_slice) + num += len(data_slice) +*/ + avrdude_message(MSG_DEBUG, "%s: ST16 to *ptr++ with RSD, data length: 0x%03X in blocks of: %d\n", progname, words * 2, blocksize); + + unsigned int temp_buffer_size = 3 + 3 + 2 + (words * 2) + 3; + unsigned int num=0; + unsigned char* temp_buffer = malloc(temp_buffer_size); + + if (temp_buffer == 0) { + avrdude_message(MSG_INFO, "%s: Allocating temporary buffer failed\n", progname); + return -1; + } + + if (blocksize == -1) { + blocksize = temp_buffer_size; + } + + temp_buffer[0] = UPDI_PHY_SYNC; + temp_buffer[1] = UPDI_STCS | UPDI_CS_CTRLA; + temp_buffer[2] = 0x0E; + temp_buffer[3] = UPDI_PHY_SYNC; + temp_buffer[4] = UPDI_REPEAT | UPDI_REPEAT_BYTE; + temp_buffer[5] = (words - 1) & 0xFF; + temp_buffer[6] = UPDI_PHY_SYNC; + temp_buffer[7] = UPDI_ST | UPDI_PTR_INC | UPDI_DATA_16; + + memcpy(temp_buffer + 8, buffer, words * 2); + + temp_buffer[temp_buffer_size-3] = UPDI_PHY_SYNC; + temp_buffer[temp_buffer_size-2] = UPDI_STCS | UPDI_CS_CTRLA; + temp_buffer[temp_buffer_size-1] = 0x06; + + if (blocksize < 10) { + if (updi_physical_send(pgm, temp_buffer, 6) < 0) { + avrdude_message(MSG_INFO, "%s: Failed to send first package\n", progname); + free(temp_buffer); + return -1; + } + num = 6; + } + + while (num < temp_buffer_size) { + int next_package_size; + + if (num + blocksize > temp_buffer_size) { + next_package_size = temp_buffer_size - num; + } else { + next_package_size = blocksize; + } + + if (updi_physical_send(pgm, temp_buffer + num, next_package_size) < 0) { + avrdude_message(MSG_INFO, "%s: Failed to send package\n", progname); + free(temp_buffer); + return -1; + } + + num+=next_package_size; + } + free(temp_buffer); + return 0; +} + int updi_link_repeat(PROGRAMMER * pgm, uint16_t repeats) { /* diff --git a/avrdude/updi_link.h b/avrdude/updi_link.h index e6848b0d..5b5e5bda 100644 --- a/avrdude/updi_link.h +++ b/avrdude/updi_link.h @@ -42,6 +42,7 @@ int updi_link_ld_ptr_inc(PROGRAMMER * pgm, unsigned char * buffer, uint16_t size int updi_link_ld_ptr_inc16(PROGRAMMER * pgm, unsigned char * buffer, uint16_t words); int updi_link_st_ptr_inc(PROGRAMMER * pgm, unsigned char * buffer, uint16_t size); int updi_link_st_ptr_inc16(PROGRAMMER * pgm, unsigned char * buffer, uint16_t words); +int updi_link_st_ptr_inc16_RSD(PROGRAMMER * pgm, unsigned char * buffer, uint16_t words, int blocksize); int updi_link_repeat(PROGRAMMER * pgm, uint16_t repeats); int updi_link_read_sib(PROGRAMMER * pgm, unsigned char * buffer, uint16_t size); int updi_link_key(PROGRAMMER * pgm, unsigned char * buffer, uint8_t size_type, uint16_t size); diff --git a/avrdude/updi_readwrite.c b/avrdude/updi_readwrite.c index eecfb7f0..e304a510 100644 --- a/avrdude/updi_readwrite.c +++ b/avrdude/updi_readwrite.c @@ -312,9 +312,5 @@ int updi_write_data_words(PROGRAMMER * pgm, uint32_t address, uint8_t * buffer, avrdude_message(MSG_INFO, "%s: ST_PTR operation failed\n", progname); return -1; } - if (updi_link_repeat(pgm, size >> 1) < 0) { - avrdude_message(MSG_INFO, "%s: Repeat operation failed\n", progname); - return -1; - } - return updi_link_st_ptr_inc16(pgm, buffer, size); + return updi_link_st_ptr_inc16_RSD(pgm, buffer, size >> 1, -1); }