From b49b07cb7596eb4f4c690fb12063f3773ebe781e Mon Sep 17 00:00:00 2001 From: Dave Marples Date: Tue, 12 Nov 2019 09:05:25 -0600 Subject: [PATCH] rch/arm/src/imxrt/imxrt_serial.c: o, just to complete the documentation of this...it looks like two issues, which confused me greatly; 1) Operational issue TC (Transmission complete) and TDRE (TX Buffer Empty) were transposed in imxrt_serial.c. The end result was that for unoptimised code everything worked fine, but optimised code got itself into a real mess and continually fired interrupts. The patch attached fixes this. This one would have been found much more quickly if this particular board had supported SWO :-/ 2) Startup issue There are a number of chip errata that apply to the 1052 first revision (A-suffix) that don't apply to the second (B-suffix). Those got me for a while and it's important to use an EVK_B_ dev board if you're suffering stability problems with this particular chip. However, even with that resolved with either optimised or unoptimised code when there is no SNVS (Battery Backup) power and the power is switched on the CPU appears to boot but gets stuck with timer interrupts not being generated. The CPU is running and it will execute linear code. I have determined this by putting an 'imxrt_lowputc('A'+irq)' into up_doirq. For the non-running case the output looks like this; LLLL this implies 4 0x0b interrupts have been generated, and nothing else. If I hit keys on the keyboard I get 'e' in the flow. 0x0b is the SVC instruction and is the mechanism by which NuttX handles task switching. 'e' is 0x24, which corresponds to the LPUART1 interrupt. Other than these, the system does not respond further but is happily in the idle loop. In this circumstance if you hit SW2 on the EVKB board though the logjam 'unjams' and normal service is established, output now looks like; LLLLPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPPLPP ...etc. Where 'P' is 0x0f (the SYSTICK interrupt)....this is what _should_ be happening so; SYSTICK starts up after SW2 has been pressed. From this point on everything works correctly and you can reboot the CPU, put new code into it or do whatever you wish, **provided power is not cycled off**. If power is cycled off then return to the top and go through the process again. When there is SNVS power to the CPU then power can be switched on and off as you wish, and the CPU will boot correctly. This has been determined by putting 3V onto J6. I think we are probably doing something naughty with the way we are starting the clocks to the timers. There are certainly some restrictions on imxrt clock manipulation which we have studiously avoided. I am investigating further and have some potential patches but even if I find the root cause it should not be included in 8.2 as the change could be disruptive. --- arch/arm/src/imxrt/imxrt_serial.c | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/arch/arm/src/imxrt/imxrt_serial.c b/arch/arm/src/imxrt/imxrt_serial.c index 6a2842dc98..aa2443d7ce 100644 --- a/arch/arm/src/imxrt/imxrt_serial.c +++ b/arch/arm/src/imxrt/imxrt_serial.c @@ -1492,7 +1492,7 @@ static void imxrt_txint(struct uart_dev_s *dev, bool enable) * Name: imxrt_txready * * Description: - * Return true if the transmit is completed + * Return true if the transmit register is available to be written to * ****************************************************************************/ @@ -1502,14 +1502,14 @@ static bool imxrt_txready(struct uart_dev_s *dev) uint32_t regval; regval = imxrt_serialin(priv, IMXRT_LPUART_STAT_OFFSET); - return ((regval & LPUART_STAT_TC) != 0); + return ((regval & LPUART_STAT_TDRE) != 0); } /**************************************************************************** * Name: imxrt_txempty * * Description: - * Return true if the transmit reg is empty + * Return true if the transmission has completed and been sent to line. * ****************************************************************************/ @@ -1519,7 +1519,7 @@ static bool imxrt_txempty(struct uart_dev_s *dev) uint32_t regval; regval = imxrt_serialin(priv, IMXRT_LPUART_STAT_OFFSET); - return ((regval & LPUART_STAT_TDRE) != 0); + return ((regval & LPUART_STAT_TC) != 0); } /****************************************************************************