-
Notifications
You must be signed in to change notification settings - Fork 988
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[RCFC] LPC4322 SWD-SGPIO. #907
base: develop
Are you sure you want to change the base?
Conversation
v2:
|
v3:
v2:
|
// SHIFT_CLOCK = DAP_Data.nominal_clock | ||
static uint32_t shift_clk_preset() | ||
{ | ||
return SystemCoreClock/DAP_Data.nominal_clock - 1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This should be using PLL1. This is how I've seen it used in the sgpio_uart NXP's Application Notes.
At the moment I'm trying to make CGU aware of CGU_PERIPHERAL_SGPIO, which is mentioned in the User Manual but
nowhere to be found in lpc43xx_cgu.c.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done. using PLL0AUDIO
//////////////////////////////////////// | ||
//////////////////////////////////////// | ||
// handle DAP_TRANSFER_FAULT | ||
// check parity |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Add parity check and return DAP_TRANSFER_ERROR on mismatch.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done.
|
||
// bits IN -> SLICE_H -> SLICE_P | ||
static void sequence_read(uint32_t bits_sz, uint8_t *data) | ||
{ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Proper implementation is needed on this one.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done.
v4: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you @groleo this is awesome. I tried your code on LPC-Link2 (LPC4370) and it was fun to tinker around with SGPIO peripheral.
So far I've found 3 concerns. 2 of them are rather trivial. please see my suggested changes.
The real problem I'm experiencing is that I can't run it slowly, which is unexpected.
When SGPIO_CLK is at 120 MHz it runs no problem at SWD clock rate of 30 MHz. like you have explained. And it seems it's fine to lower the clock rate down to 3 MHz. But beyond that it starts to fail. Using non SGPIO version of DAPLink it can go just down to 1 kHz .
SLICE_POS(SLICE_SWDIO_TXEN) = POS_POS(period) | POS_PRESET(0); | ||
SLICE_REG(SLICE_SWDIO_TXEN) = 0b00000000000000000000000000000000; | ||
SLICE_SHD(SLICE_SWDIO_TXEN) = 0; | ||
SLICE_CNT(SLICE_SWDIO_TXEN) = 0; | ||
|
||
SLICE_POS(SLICE_SWDIO_TMS_OE) = POS_POS(period) | POS_PRESET(0); | ||
SLICE_REG(SLICE_SWDIO_TMS_OE) = 0b00000000000000000000000000000000; | ||
SLICE_SHD(SLICE_SWDIO_TMS_OE) = 0; | ||
SLICE_CNT(SLICE_SWDIO_TMS_OE) = 0; | ||
SLICE_REG(SLICE_SWDIO_TMS_DOUT0) = 0; | ||
SLICE_SHD(SLICE_SWDIO_TMS_DOUT0) = 0; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SLICE_POS(SLICE_SWDIO_TXEN) = POS_POS(period) | POS_PRESET(0); | |
SLICE_REG(SLICE_SWDIO_TXEN) = 0b00000000000000000000000000000000; | |
SLICE_SHD(SLICE_SWDIO_TXEN) = 0; | |
SLICE_CNT(SLICE_SWDIO_TXEN) = 0; | |
SLICE_POS(SLICE_SWDIO_TMS_OE) = POS_POS(period) | POS_PRESET(0); | |
SLICE_REG(SLICE_SWDIO_TMS_OE) = 0b00000000000000000000000000000000; | |
SLICE_SHD(SLICE_SWDIO_TMS_OE) = 0; | |
SLICE_CNT(SLICE_SWDIO_TMS_OE) = 0; | |
SLICE_REG(SLICE_SWDIO_TMS_DOUT0) = 0; | |
SLICE_SHD(SLICE_SWDIO_TMS_DOUT0) = 0; | |
// contents of registers are directly reflected to corresponding signals. | |
SLICE_REG(SLICE_SWDIO_TXEN) = 0b00000000000000000000000010000000; | |
SLICE_REG(SLICE_SWDIO_TMS_OE) = 0b00000000000000000000000000000001; | |
SLICE_REG(SLICE_SWDIO_TMS_DOUT0) = 0; | |
if (info & SWD_SEQUENCE_DIN) { | ||
sequence_read(bytes_nb*8, swdi); | ||
} else { | ||
sequence_send(bytes_nb*8, swdo); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
if (info & SWD_SEQUENCE_DIN) { | |
sequence_read(bytes_nb*8, swdi); | |
} else { | |
sequence_send(bytes_nb*8, swdo); | |
} | |
if (info & SWD_SEQUENCE_DIN) { | |
sequence_read(bytes_nb, swdi); | |
} else { | |
sequence_send(bytes_nb, swdo); | |
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
sequence_send() takes a bits_sz (bits size) as first parameter. I can replace the 8 to BITS_IN_CHAR to clarify the intent.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
info & SWD_SEQUENCE_CLK
specifies clock cycles , so it is bits , not bytes . I could have renamed bytes_nb to bits_nb . anyway naming is up to you.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Now i get what you mean (thanks for taking a good look at this).
I'll change bytes_nb to bits_nb.
SLICE_SHD(SLICE_SWDIO_TXEN) = 0; | ||
|
||
uint32_t bits_per_chunk = bits_sz<=64 ? bits_sz : 64; | ||
SLICE_POS(SLICE_SWDIO_TMS_DIN1) = POS_POS(bits_per_chunk) | POS_PRESET(bits_per_chunk); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SLICE_POS(SLICE_SWDIO_TMS_DIN1) = POS_POS(bits_per_chunk) | POS_PRESET(bits_per_chunk); |
|
||
if (bits_sz <= BYTES_PER_SLICE*8) | ||
{ | ||
slices_used = SLICE_SWDIO_TMS_DIN0; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
slices_used = SLICE_SWDIO_TMS_DIN0; | |
slices_used = SLICE_SWDIO_TMS_DIN0_MASK; |
} | ||
else | ||
{ | ||
slices_used = SLICE_SWDIO_TMS_DIN0 | SLICE_SWDIO_TMS_DIN1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
slices_used = SLICE_SWDIO_TMS_DIN0 | SLICE_SWDIO_TMS_DIN1; | |
slices_used = SLICE_SWDIO_TMS_DIN0_MASK | SLICE_SWDIO_TMS_DIN1_MASK; |
uint32_t bits_remaining = bits_sz - bits_it; | ||
uint32_t bytes_remaining = (bits_remaining + 7) / 8; | ||
uint32_t slice_it = 0; | ||
uint32_t slices_used = 0; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
uint32_t slices_used = 0; |
// Disable TXEN. | ||
SLICE_POS(SLICE_SWDIO_TXEN) = POS_POS(0) | POS_PRESET(0); | ||
SLICE_CNT(SLICE_SWDIO_TXEN) = 0; | ||
SLICE_REG(SLICE_SWDIO_TMS_OE) = 0; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SLICE_REG(SLICE_SWDIO_TMS_OE) = 0; | |
SLICE_REG(SLICE_SWDIO_TXEN) = 0; |
I'm experimenting reconfiguration of PLL to match required SWD frequency. Based on your code. It can run at SWD clock speed ranging 50MHz down-to 1kHz . |
Signed-off-by: Adrian Negreanu <[email protected]>
Current status of this pull-request.
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wasn't sure until now if this change is needed or not.
Now I had a chance to test with and without this change, and I'm convinced that this change fixes stability when connecting with some target , ex nRF52840 .
#define SGPIO_SWDIO_TXEN 15 | ||
#define SLICE_SWDIO_TXEN SLICE_M |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
#define SGPIO_SWDIO_TXEN 15 | |
#define SLICE_SWDIO_TXEN SLICE_M | |
#define SGPIO_SWDIO_TXEN 15 | |
#define SGPIO_SWDIO_TXEN_DOUTM8_BIT 7 | |
#define SLICE_SWDIO_TXEN SLICE_M |
uint32_t parity; | ||
|
||
SLICE_POS(SLICE_SWDIO_TXEN) = POS_POS(8) | POS_PRESET(0); | ||
SLICE_REG(SLICE_SWDIO_TXEN) = 0b00000000000000001111111111111111; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SLICE_REG(SLICE_SWDIO_TXEN) = 0b00000000000000001111111111111111; | |
SLICE_REG(SLICE_SWDIO_TXEN) = 0b11111111 << SGPIO_SWDIO_TXEN_DOUTM8_BIT; |
uint32_t pkt[2]; | ||
|
||
SLICE_POS(SLICE_SWDIO_TXEN) = POS_POS(8) | POS_PRESET(0); | ||
SLICE_REG(SLICE_SWDIO_TXEN) = 0b00000000000000001111111111111111; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SLICE_REG(SLICE_SWDIO_TXEN) = 0b00000000000000001111111111111111; | |
SLICE_REG(SLICE_SWDIO_TXEN) = 0b11111111 << SGPIO_SWDIO_TXEN_DOUTM8_BIT; |
slice_pos_disable(SLICE_SWDIO_TMS_DOUT0_MASK | SLICE_SWDIO_TMS_DOUT1_MASK | SLICE_SWCLK_TCK_MASK | SLICE_SWDIO_TXEN_MASK | SLICE_SWDIO_TMS_OE_MASK); | ||
|
||
SLICE_REG(SLICE_SWDIO_TMS_OE) = 1; | ||
SLICE_REG(SLICE_SWDIO_TXEN) = 1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SLICE_REG(SLICE_SWDIO_TXEN) = 1; | |
SLICE_REG(SLICE_SWDIO_TXEN) = 1 << SGPIO_SWDIO_TXEN_DOUTM8_BIT; |
slice_pos_disable(SLICE_SWDIO_TMS_DIN0_MASK | SLICE_SWDIO_TMS_DIN1_MASK | SLICE_SWCLK_TCK_MASK); | ||
|
||
SLICE_REG(SLICE_SWDIO_TMS_OE) = 1; | ||
SLICE_REG(SLICE_SWDIO_TXEN) = 1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SLICE_REG(SLICE_SWDIO_TXEN) = 1; | |
SLICE_REG(SLICE_SWDIO_TXEN) = 1 << SGPIO_SWDIO_TXEN_DOUTM8_BIT; |
I want to see this going forward. |
Hi @elfmimi , what testing infra do you use? I've more or less used the on-board debug probe (on the rt1170-evk) I also have a OM13070 (which has a LPC4322) thx |
Oh, I used OM13084 (interface: LPC4322 , target: LPC43S67) . In case of OM13070 , LPCX4337_V3_SCHEMATIC_REVA3 , you can access SWD pins of LPC4322 through these test points. For logging/output Segger RTT might be the best option. |
Please have a look at this branch for how to do RTT. @groleo After executing the above code, attach with pyocd to get the log. |
Cool. |
Yes, it's cool! No. SWD is SWD. bring in another CMSIS-DAP adapter. |
which probe should I get ? |
picoprobe |
that "may be ok" sounds worrisome (i'd like to avoid debugging the debugger). Do you have a recommendation that's sure to work (or you use yourself) ? |
I suppose we should move this talk to daplink channel of pyOCD slack. |
Requires #906.
This p-r is replacing the bit-banged SWD implementation with one which uses the lpc4322's SGPIO.
Transfer rate reported by IAR is around 380 kB/sec with DAP.nominal_clock @30mhz and SGPIO_CLK@120MHz.
Maximum I could get was 420 kB/sec with DAP.nominal_clock @40MHz and SGPIO_CLK @204MHz;
testing in progress.