/atomicx

Pure C++ non stack displacement that implements cooperative multitask library for SINGLE CORE embedded development on DSPs, Microcontrollers and Processor (ARV, RISCV, ARM(all), TENSY, ESP), while also suitable for applications on Windows, Linux and MacOs and compatible with some RTOSs as well. This library allows full event driven applications while uses SMARTs LOCKS and WAIT/NOTIFY locks to also transport messages, MESSAGE BROKER is also provided (Those uses Message type size_t message and size_t tags, where tag will give meaning to the message). That implementation also introduce thread safe QUEUE (full object) and smart_ptr (to allow better implementation on minimal environment)

Primary LanguageC++MIT LicenseMIT

AtomicX

Version 1.2.1 release

image

What is AtomicX? AtomicX is a general purpose cooperative thread lib for embedded applications (single core or confined within other RTOS) that allows you partition your application "context" (since core execution) into several controlled context using cooperative thread. So far here nothing out of the ordinary, right? Lets think again:

Backlog and updates

Implementations from Work on progress

Version 1.3.0

  • NOW, I am thrilled to announce that AtomicX have send and receive functions that enables data transferring between two or more threads, allowing real client / server application strategy inside embedded and non-embedded application, this way, small MCUs can be used to easily transport stack data (which is protected in this thread system) to other thread.

  • Ported and enhanced the DotMatrix project, it implements a full Dot Led Matrix scrolling text system with

    • Serial terminal
    • Telnet Terminal
    • UDP Trap (You can send a UDP message to IP/2221 and it will display, I deal for trapping messages)
    • An amazing general log API based on iostream, capable of adding "Specialized Loggers", and add using logger.AddLogger:
          logger << LOG::ERROR << "Failed to start WiFi." << std::endl;
    • based on ESP8266 and 8 or 4 Dot Matrix leds array
    • Just connect the Led Matrix (Dot Matrix Modul 8x8 Display Matrix Max7219 Led Lcd with 8 or 4 8x8 display)
        Designed for NodeMCU ESP8266
        ################# DISPLAY CONNECTIONS ################
        LED Matrix Pin -> ESP8266 Pin
        Vcc            -> 3v  (3V on NodeMCU 3V3 on WEMOS)
        Gnd            -> Gnd (G on NodeMCU)
        DIN            -> D7  (Same Pin for WEMOS)
        CS             -> D4  (Same Pin for WEMOS)
        CLK            -> D5  (Same Pin for WEMOS)
    
  • Added WaiTAny, that extends the Wait/Notify functionality, since it will also receive, and return by reference, ANY TAG. giving the developer ability to easily create a full Client / Service infra structure.

  • Dropping BROKER functionality, and welcoming Broadcasting functionality, it will enable a thread to receive all broadcasts asynchronously sent by other threads, only by enabling it and implementing the handler :

        virtual void BroadcastHandler (const size_t& messageReference, const Message& message) 

    and enabling it using the code:

        SetReceiveBroadcast (true);

    The handler will deliver tree parameters:

    • size_t MessageReference that works like a reference of what is the message about;
    • size_t Message.message which is the message payload (could be even an pointer);
    • size_t Message.tag that can be used as a meaning for the message:

    Example:

    • messageReference=BROADCAST_IRCAMERA
    • Message.message=CAMERA_DONE
    • Message.tag=CAMERA_READINGS

    On this simple example using 'mnemonics', that could have been enum class or directives, a single message was able to inform asynchronously that a IR CAMERA just read something, but could have informed ERROR or even that it was READING...., the approach allows a powerful controller, since you can apply layers on your code making processing really fast and precise and less messages will travel across the systems.

  • mutex and smartMutex now have timeout, by using lock(<timeout time>) and sharedLock(<timeout time>), if no timeout is given: lock() or sharedLock() wait indefinitely (fully back compatible with existing code)

Version 1.2.1

  • Adding Dynamic Nice, now it is possible to let the kernel set the best performance for your thread, for this SetNice(*initial nice*) and than SetDynamicNice(true) in the constructor of your thread. The kernel will be able to always adjust your thread for Best performance, but, it will leave no room for sleeps between threads, increasing power consumption, it is powerful but use it carefully.

  • Added YieldNow() the higher priority context change, it will allow other threads to work, but will, also return faster than others

  • smartSemaphore, Used to compliance with RAII, once used in the thread context, it takes a semaphore to be initialized and expose the same methods, although it manages the local context, and ones it it gets out of context, due to leaving {} or a functions, for example the semaphore shared context is released if ever taken during the smartSemaphore instantiated object life cycle. The same is available for mutex, called smartMutex, follows the same principle.

  • IMPORTANT, Introducing Semaphores, atomicx::semaphore(<How many shared>), now you can use methods (acquire() or acquire(timeout)) and release() along with GetCount, GetMaxAcquired, GetWaitCount and static method GetMax to return the maximum shared you can use to instantiate. Examples for Arduino and PC where also introduced and fully tested.

  • Introducing atomicx::Timeout, this will help tracking a timeout over time, using methods IsTimedout and GetRemaining and GetDurationSince. Special use case, if the timeout value is zero, IsTimedout will always return false.

  • IMPORTANT NOTIFICATION atomicx::lock has been renamed to atomicx::mutex for consistency, all methods are the same.

  • Improvement Added a contructor for self-manager start to define a start size and increase pace. For example: a thread starts with 150 bytes and increase pace of 10, but used stack was 200, the kernel will do 200 + 10 (increase pace) to give it room to work. The default value is (1)

        /**
         * @brief Construct a new atomicx object and set initial auto stack and increase pace
         *
         * @param nStackSize            Initial Size of the stack
         * @param nStackIncreasePace    defalt=1, The increase pace on each resize
         */
        atomicx(size_t nStackSize, int nStackIncreasePace=1);

Version 1.2.0

  • INTRODUCING Self managed stack, now it is possible to have self-managed stack memory for any threads, no need to define stack size... (although use it with care) just by not providing a stack memory, AtomicX will automatically switch the tread to self-managed, to do just use atomicx() default constructor instead.

    Notes: * It will only entries the stack enough to hold what is needed if the used stack is greater than the stack memory managed. * No decrease of the stack size was added to this release. * In case your thread is not able to resize the stack, if it needs more, StackOverflowHandle is called.

Examples: - Ardunino/Simple - avrAutoRobotController

  • Explicitly added the pc example shown here to to examples/pc as simple along with makefile for it. Also updated it to have an example of Self-managed stack memory as well.

Version 1.1.3

  • Added a Thermal Camera Demo ported from CorePartition but now fully object oriented

  • POWERFUL: Now Wait`Notify` will accept a new parameter called subType, the name gives no clue but it is really powerfull it allows developer to create custom Types of notifications, that same strategy is used when syncNotify is called and get blocked until a timeout occur or a wait functions is used by another thread.

Version 1.1.2

  • *Important Notify was split into Notify and SyncNotify to avoid compilation ambiguity reported for some boards, all the examples have been migrated to use one of those accordingly and tested against all supported processors.

Version 1.1.1

  • PLEASE NOTE No Spin Lock what so ever in this Kernel, it is working fully based on Notification event along with message transportation.

  • NOTIFY are now able to sync, if a atomicx_time is provided, Notify will wait for a specific signal to inform a Wait for refVar/Tag is up. This is a important feature toward using WAIT/Notify reliably, while your thread can do other stuffs on idle moment

  • avrRobotController simulator for Arduino, is introduced, to show real inter process communication, it will open a terminal and both commands are available: system - To show Memory, Threads and motor status and move <flot motor A> <flot motor B> <flot motor C>

Version 1.1.0

  • finish() method will be call every time run() is returned, this allow special cases like eventual threads to self-destroy itself, otherwise the object would be only a memory leak.... see examples on main.cpp

  • smartMutex RAII (Resource Acquisition Is Initialization) compliance, allow mutex or shared mutex to be auto release on object destruction.

  • IMPORTANT Now Notifications (Wait/Notify) can be timedout. if Tick based time is given, the waiting procedure will only stay blocked during it. (NO SPIN LOCK, REAL STATE BLOCK)

  • IMPORTANT LookForWaitings block for timeout time will a wait for specific refVar/tag is available, otherwise timeout, can be used sync wait and notify availability

  • IMPORTANT Now Wait/Notify Tags, used to give meaning/channel to a notification can be se to "all tags" if Tag is zero, otherwise it will respect refVar/Tag

Version 1.0.0

  • DOES NOT DISPLACE STACK, IT WILL STILL AVAILABLE FOR PROCESSING, the Stack Page will only hold a backup of the most necessary information needed, allowing stacks in few bites most if the time. This implementation if highly suitable for Microcontrollers like ATINY85, for example, that only has 512 bites, and you can have 5 or more threads doing things for you, only backup the most important context information.

  • STACK MEMORY ARE PROTECTED by nature and confined to the context execution only, if you want to exchange non-data global use atomicx::Send and atomicx::Receive.

    • IMPORTANT: DO NOT USE CONTEXT(stack) MEMORY POINTER to exchange information to other threads, extractables like wait/notify, lock uses global memory instead. All threads will use the default stack memory which is protected during execution, instead use Global variables, allocated memory or atomicx_smart_ptr objects. Alternatively use atomicx::send and atomicx::receive to transport stack, global or heap memory data.
  • Since it implements Cooperative thread every execution will atomic between atomicx thrteads.

  • AtomicX DOES NOT DISPLACE STACK, yes, it will use a novel technique that allow you to use full stack memory freely, and once done, just call Yield() to switch the context.

    1. Allow you to use all your stack during thread execution and only switch once back to an appropriate place
        Stack memory
        *-----------*
        |___________| Yield()
        |___________|    thread 0..N
        |___________|     |       .  - After context execution
        |___________|     |      /|\   is done, the developer can
        |___________|     |       |    choose where to switch
        |___________|     |       |    context, saving only what is
        |___________|    \|/      |    necessary
        |___________|     ---------
        |           |     - During context
        *-----------*       can goes deeper as
                            necessary
    
  • Due to the zero stack-displacement technology, developers can ensure minimal stack memory page, allowing ultra sophisticated designes and execution stack diving and only backing up to the stack memory page what is necessary.

  • Full feature for IPC (Inter Process Communication)

    • Thread safe Queues for data/object transporting.
    • EVERY Smart Lock can transport information (atomicx::message)
    • Message is composed by "size_t atomix::message and a "size_t tag"
      • This novel concept of "tag"s for an atomicx::message gives the message meaning.
      • Since atomicx::message uses size_t messages can also transport pointers
    • Smart Locks can Lock and Shared Lock in the same object, making
    • Full QUEUE capable to transport objects.
  • Full feature for IPN (Inter Process Notification)

    • Thread can wait for an event to happen.
    • On event notification a atomix::message can be sent/received
  • (DEPRECATED) A message broker based on observer pattern (NOW DROPPED and REPLACED BY BROADCAST)

    • A thread can use WaitBroker Message to wait for any specifc topic asynchronously.
    • Instead of having a Subcrib call, the developer will provide a IsSubscribed method that the kernel will use to determine if the object/thread is subscribed to a given topic.
    • Broker uses atomicx::message to transport information. For inter process Object transport, please use atomicx::queue.
  • ALL WAIT actions will block the thread, on kernel level (setting thread to a waiting state), until the notification occurs. Alternatively the notification can be transport a atomicx::message structure (tag/message)

    • WAIT and NOTIFY (one or all) will use any pointer as the signal input, virtually any valid address pointer can be used. IMPORTANT: Unless you know what you are doing, do NOT use context pointer (execution stack memory), use a global or allocated memory instead (including atomicx::smart_prt)
  • All Notifications or Publish functions will provide a Safe version, that different from the pure functions, will not trigger a context change and the function will only fully take effect onces the context is changed in the current thread where the interrupt request happened.

  • IMPORTANT since all threads will be executed in the "default" stack memory, it will not be jailed in the stack size memory page, DO NOT USE STACK ADDRESS TO COMMUNICATE with another threads, use only global or alloced memory pointers to communicate

  • IMPORTANT In order to operate with precision, specialise ticks by providing either atomicx_time Atomicx_GetTick (void) and void Atomicx_SleepTick(atomicx_time nSleep) to work within the timeframe (milleseconds, nanoseconds, seconds.. etc). Since AtomicX, also, provice, Sleep Tick functionality (to handle idle time), depending on the sleep time, to developer can redude the processor overall consuption to minimal whenever it is not necessary.

    • Since it will be provided by the developer, it gives the possibility to use external clocks, hardware sleep or lower consumptions and fine tune power and resource usages.

    • If not specialization is done, the source code will use a simple and non-deterministic loop cycle to count ticks.

//
//  main.cpp
//  atomicx
//
//  Created by GUSTAVO CAMPOS on 28/08/2021.
//

#include <unistd.h>
#include <sys/time.h>
#include <unistd.h>

#include <cstring>
#include <cstdint>
#include <iostream>
#include <setjmp.h>
#include <string>

#include "atomicx.hpp"

using namespace thread;

#ifdef FAKE_TIMER
uint nCounter=0;
#endif

void ListAllThreads();

/*
 * Define the default ticket granularity
 * to milliseconds or round tick if -DFAKE_TICKER
 * is provided on compilation
 */
atomicx_time Atomicx_GetTick (void)
{
#ifndef FAKE_TIMER
    usleep (20000);
    struct timeval tp;
    gettimeofday (&tp, NULL);

    return (atomicx_time)tp.tv_sec * 1000 + tp.tv_usec / 1000;
#else
    nCounter++;

    return nCounter;
#endif
}

/*
 * Sleep for few Ticks, since the default ticket granularity
 * is set to Milliseconds (if -DFAKE_TICKET provide will it will
 * be context switch countings), the thread will sleep for
 * the amount of time needed till next thread start.
 */
void Atomicx_SleepTick(atomicx_time nSleep)
{
#ifndef FAKE_TIMER
    usleep ((useconds_t)nSleep * 1000);
#else
    while (nSleep); usleep(100);
#endif
}

/*
 * Object that implements thread with self-managed (dynamic) stack size
 */
class SelfManagedThread : public atomicx
{
public:
    SelfManagedThread(atomicx_time nNice) : atomicx()
    {
        SetNice(nNice);
    }

    ~SelfManagedThread()
    {
        std::cout << "Deleting " << GetName() << ": " << (size_t) this << std::endl;
    }

    void run() noexcept override
    {
        size_t nCount=0;

        do
        {
            std::cout << __FUNCTION__ << ", Executing " << GetName() << ": " << (size_t) this << ", Counter: " << nCount << std::endl << std::flush;

            nCount++;

        }  while (Yield());

    }

    void StackOverflowHandler (void) noexcept override
    {
        std::cout << __FUNCTION__ << ":" << GetName() << "_" << (size_t) this << ": needed: " << GetUsedStackSize() << ", allocated: " << GetStackSize() << std::endl;
    }

    const char* GetName (void) override
    {
        return "Self-Managed Thread";
    }
};

/*
 * Object that implements thread
 */
class Thread : public atomicx
{
public:
    Thread(atomicx_time nNice) : atomicx(stack)
    {
        SetNice(nNice);
    }

    ~Thread()
    {
        std::cout << "Deleting " << GetName() << ": " << (size_t) this << std::endl;
    }

    void run() noexcept override
    {
        size_t nCount=0;

        do
        {
            std::cout << __FUNCTION__ << ", Executing " << GetName() << ": " << (size_t) this << ", Counter: " << nCount << std::endl << std::flush;

            nCount++;

        }  while (Yield());

    }

    void StackOverflowHandler (void) noexcept override
    {
        std::cout << __FUNCTION__ << ":" << GetName() << "_" << (size_t) this << ": needed: " << GetUsedStackSize() << ", allocated: " << GetStackSize() << std::endl;
    }

    const char* GetName (void) override
    {
        return "Thread";
    }

private:
    uint8_t stack[1024]=""; //Static initialization to avoid initialization order problem
};


int main()
{
    Thread t1(200);
    Thread t2(500);

    SelfManagedThread st1(200);

    // This must creates threads and destroy on leaving {} context
    {
        Thread t3_1(0);
        Thread t3_2(0);
        Thread t3_3(0);

        // since those objects will be destroied here
        // they should never start and AtomicX should
        // transparently clean it from the execution list
    }

    Thread t4(1000);

    atomicx::Start();
}