-
Notifications
You must be signed in to change notification settings - Fork 66
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
Add/mix/graft STM8 eForth into a C project #235
Comments
Interesting problem - I had that on my list for a very long time :-) The TSDZ2 has a Medium Density (STM8S105x4) device with 2K RAM and 16K/32K Flash. As long as the C application can live with, I'd say, 1K RAM the 4-5K Flash needed for the Forth shouldn't be a big problem. Low Density devices with 1K RAM and 8K Flash, like STM8S003F3P6, would only allow a very frugal C application. It's easiest if the Forth application is in the foreground, and whatever C there is lives in interrupts. Obvious use cases are:
What needs to be done to getting C code into the foreground depends on the use case:
Can you think of other use-cases or applications? |
In the case of the ebike display I can't see why the main loop shouldn't run in a low-priority interrupt with 10ms cadence. In the use cases above I forgot to mention the trivial case where C functions are of the type |
SDCC passes parameters on the return stack (or in registers) and leaves the stack set-up and the clean-up to the caller. Good for the Forth interface. There is an example of how to implement C functions in assembly here but the STM8 parameter passing scheme in SDCC is largely in the dark). At first glance it looked like it's not so simple but with the help of some experiments it turns out that it's not complicated at all: The simple function
Hypothesis:
Note that returning structs ("aggregates") isn't implemented in SDCC Version 3.6.0 #9615 which I used for tests ( Based on the hypothesis above writing wrappers with Passing C symbols as addresses into Forth is a different question. Maybe it's easier to write a script that creates a wrapper at C-compile-time ;-) |
I made a new tg951/docker-sdcc:latest using SDCC 3.8.4 #10751. So far |
|
@sbridger sorry for the late reply - the flu still has me. I'd like rephrase your proposals it (for fun and profit): 1st: use Forth as some kind of a DSL for a highly variable system concern (the display that everyone wants different.) 2nd: use Forth as some kind of a DSL for customized system features that have a different life cycle than the underlying implementation 3rd: use Forth as a library for improved code density inside C as a size/engineering-effort trade-off Can you think of an existing C showcase application? The Open-Source-EBike-Firmware is very interesting but it's a bit expensive to build a test set-up. |
Not off the top of my head for STM8. On ARM or 8051, I would say BLE modules. The bluetooth libaries are all C, but I think that a forth programmable BLE/bluetooth module would be rather nice. They have heaps of either flash or ram code space for 4th. They also tend to have a bit of a big ugly C codebase, and for CC2541 a commercial compiler, which is all an overkill for making a simple BLE gadget. |
I agree that complex SoC with an opaque code base have the most benefit from a lightweight system-integration oriented language. A good example is Punyforth for the ESP8266. Perhaps I'll try something from STM8_templates by @gicking. It's not yet clear to me how well it's integrated with SDCC, and I guess it would be best to pull everything into a Docker container. Edit: after initial tests and discussions with @gicking sduino appears to be a better choice. |
As a matter of interest, how big is the assembler overhead of using the (4th) data stack for params of a C function call, instead of the return stack? |
There is quite some overhead: Data Stack push/pull on the STM8 is quite expensive: there is no index register auto-increment. In STM8 eForth the inefficient push/pull is hidden in the Forth primitives. Apparently there are Forth systems that use DTC (i.e. no CALL/RET) that use the return stack as the data stack. Such a system would be easier to integrate with SDCC but it would have other trade-offs. On the other hand, C has a self inflicted penalty: the parameters placed on the return stack before the call must be removed after the call! Data passing between functions is therefor much more expensive than in Forth. Some µC C compilers (e.g. Keil MCS51) try hard to do all parameter passing through registers, which sometimes leads to efficient chaining of functions. That works quite well for flat call hierarchies but it has obvious limitations (don't even think about recursion). The main problem is that data has to be copied because there is no "contract" in C that a function are allowed to alter data on stack of he calling function. Forth is much more akin to languages with nested functions. Functional languages with closures are on a next level. In Forth, this would require to pass stacks around. |
Issue #235 compatibility with SDCC 3.8.4
Merge d2c29ea makes STM8 eForth compatible with SDCC 3.8.4. Daniel Drotos, the author of uCsim, of which sstm8 is used in simload.sh and codeload.py, implemented a new "lockstep mode" for the telnet interface of the STM8 UART simulation. Thanks a lot! There is now a docker-sdcc:3.8.4 and Apparently the register and stack interface to SDCC functions didn't change from SDCC 3.6.0 to SDCC 3.8.4, however, it's not certain if my understanding is right, or if there are any plans to change the interface. I raised that question in the SDCC Forum. |
There is an answer from @spth, SDCC maintainer, in the forum above: my guess based on observations was right but the interface is still considered an implementation detail even if it has been stable for a long time (I proposed to give the interface some kind of a formal status). There is also 64 bit parameter and results passing, which is maybe not the most interesting for a C-Forth interface. I updated the Wiki here. |
While looking at ebike displays, I wonder about the process to add/mix/graft e4th into a primarily C project.
Originally posted by @sbridger in #234 (comment)
The text was updated successfully, but these errors were encountered: