-
Notifications
You must be signed in to change notification settings - Fork 13
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
Support for UART1 on tiny841, 1634, and some general questions #22
Comments
Urboot already supports all UARTs. When compiling you have the options UARTNUM= and, particularly for the t441 and t841, UARTALT=1 to remap the pins. Look, for example, at the autobaud bootloaders for the t841. Fun fact: I've never tried them (don't have the part); you could be the first to try them out! See also the make options for serial comms.
urboot.c clears the MCUSR after putting a copy into R2 (fun fact: ensuring they end up in R2 costs zero bytes code). Urboot starts the application through a WDR (except after flashing through OTA), so R2 will normally see
I actually want GPIOR0 clean on app start for my IDE (not using Arduino) and don't fancy spending two bytes of code for this in the bootloader either. So that will defo be different to what your bootloaders are doing. More discussion on when to start the app and when the bootloader here. I think the way optiboot tries to keep the reset flags is over the top and believe it has whacky side effects.
Sigh, you are catching the urboot.hex repository of pre-compiled bootloaders in a time of change. Am about to change the naming scheme (it's like founding a rock group: 90% of the discussion is about the name). Long short: I want the frequency and baud rate directories on github to list in ascending order of frequency for manual selection. Issue #19 has made me aware of the intricacies of |
I noticed the |
The problem with stashing it in R2 is that you *aren't guaranteed that it will still be there by the time all the init sections have been run plus the core init functions.... There could be some innocuous looking class with a doozie of a constructor that hogs a great number of registers, and that could squeeze it out before we get to user code. The compiler has every right to use r2 as it sees fit right? So we're just hoping that the way registers are allocated for constructors and initialization code won't smash R2. I don't think that's a valid assumption. It is probably true in simple test situations, but I'm not confident it would work in a full scale system, especially not if some poorly written and inefficient arduino libraries (most of them) are thrown in.. |
Yes, all correct observations about the use of R2 for the MCUSR copy. Applications that use this feature better make use of the The urboot project specialises in small bootloaders and, as such, finds it hard to justify the use of 2 bytes code in every b/loader for stashing something in GPIOR0. The other problem is that were the bootloader to taint GPIOR0, then every application wanting to use GPIOR0 would need clear it before use: not only are another 2 bytes in those applications gone, the unsuspecting user reading the data sheet would not expect GPIOR0 to be uninitialised after reset. My application code regularly uses GPIOR0 for eight slick global Boolean variables. Here from a header file in my IDE: /*
* Provide eight global Boolean variables through GPIOR0, which normally
* reside in I/O register space, so the compiler can make use of efficient
* bit addressable cbi/sbi instructions as well as sbrs/sbrc tests. Just, eg,
*
* #define mybool0 board_bool(0)
* #define mybool1 board_bool(1)
*
* and use mybool0 and mybool1 as global Boolean variables.
*/
typedef struct { uint8_t b0:1, b1:1, b2:1, b3:1, b4:1, b5:1, b6:1, b7:1; } board_bool8_t;
#ifdef GPIOR0
#define board_bool8 GPIOR0
#define board_bool(x) (((board_bool8_t *)_SFR_MEM_ADDR(GPIOR0))-> b ## x)
#else
extern board_bool8_t board_bool8;
#define board_bool(x) (((board_bool8_t *)&board_bool8)-> b ## x)
#endif |
But anyway, seems like you have a lot of experience and have serious bootloader deployments, so I guess you can do proper field testing with these nifty little urboot bootloaders. |
Does urboot recognize that if there are no flags set, that the peripherals cannot be assumed to be in any state? On modern AVRs, this is straightforward to fix - you immediately issue a software reset (on modern AVRs, the consequences of not doing so are much worse - not only are the peripherals more complex, and hence potentially in more complicated undesired configurations, the LVL1EX bit may be set in CPUINT, because of the way BAD_ISR is defined. Since no RETI is ever executed, after a bad interrupt, even if everything else worked, it would still leave the chip in a non-functional state, because interrupts would be disabled because it thinks it's still in one, and there isn't a way to unpick this other than resetting the chip. The reason I use GPIOR0 is that getting the flags out of R2 is not something you can explain to someone how to do in 2 lines of code ( |
Urboot hands over control to the application in this case. What else do you want it to do here? It has no idea what the cause of "no flags" is. It could be
I don't see what the bootloader could (or should) do otherwise.
Urboot hasn't been extended to modern parts (yet). Might consider that when I get round to do that, but it will need some more persuasion to get the bootloader to do that. Why wouldn't the application's (probably weakly defined and therefore overridable)
You don't have to! Your core is free to put into
That way Mind you, All things considered, I still think it is a bad idea for the bootloader to use |
@SpenceKonde I guess we have discussed all you wanted to know? LMK if there are still questions unanswered (either re-open the issue or open a new one). |
So, I have heard that urboot is smaller than optiboot, and does virtual boot erase correctly. If this is the case, I would rather like to adopt this in place of Optiboot as the serial bootloader supplied with ATTinyCore. I am impressed with the variety parts you have builds for (though - I'm rather baffled by the wacky file names. though that can be normalized with a python script - the goal would be to something more like this:
urboot_attiny441_8000000UL_uart1_(more feature flags if applicable).hex
(The only operation we can do is concatenation of strings within the context of the platform definition - we have no if/then, no math, just concatenation. And so, I build the hex file names by concatenating build.fcpu, the partnumber, and parameters for options like the LED pin and what port (on the dual usart ones) and such.)
Anyway - I was wondering about a few things. So the 841 and 1634 have 2 hardware serial ports. With optiboot, I provide 3 sets of 841 binaries. One with UART0, one with UART1, and a third with UART0, but the remap register set to move the UART0 to a different set of pins. How hard would support for multiple UARTs be? Is it just a matter of building with different options?
Also, how does Urboot handle the reset flags?
My approach has (since I recognized the fatal nature of allowing execution to proceed if there are not reset flags set - I think it causes the majority of hangs that need manual intervention to correct) been to read the reset flags. If they're not 0, copy their current value to GPIOR0, and clear them in the reset flag register (you can't count on the app to do this, the app won't do this, and you;ll think you had a clean reset when in fact it was a dirty reset, and you're trying to use UART at 14400 baud and 1 MHz, except the user code turned off the div8, so now your clock is running 8 times faster than you think, so small wonder the person can't upload even though there's an Optiboot blink. The chip is running the uart at 115200 baud and 8 MHz - things like that - can happen when you smash the return address (conveniently located right at the top of the stack!), or trigger an ISR that didn't exist. while the causes are all bad bugs in their own right, you're always better off bootlooping into a working bootloader than a non-working one. So since I clear the registers, I know that if I see something there, there must have been a reset, as there always must be, so all is well, and if I see nothing, I immediately trigger a reset because that's the only thing that's likely to be successful now. Though the problem is more acute on modern AVRs ( a missing ISR lands you in a state where not only are the settings not at their reset values, everything otherwise appears to work, except that interrupts can't trigger ISRs. (You overflow an array, and then when you return, execution resumes from the start of the program but with most of the current state retained, millis doesn't advance, but micros does, though it just keeps looping through 0-9999... can you imagine trying to debug that mess? On non-bootloader configurations, I have the app do the same thing in .init3).
But yeah, I am very interested in this...
The text was updated successfully, but these errors were encountered: