These directories provide the porting layer that allows the high level C APIs to be built and run on a supported MCU.
No attempt is made to create a full MCU porting API; only the APIs necessary to run the high level C APIs are implemented here.
The api directory defines the port API, each API function documented in the header file. The clib directory contains implementations of C library APIs that may be missing on some platforms (e.g. strtok_r
, the re-entrant version of the strtok
library function). In the platform directory you will find the implementation of the porting API on various target SDKs and MCUs and the necessary instructions to create a working binary and tests on each of those target MCUs. Please refer to the README.md
files in your chosen target platform directory for more information.
The test directory contains tests for this API that can be run on any platform.
If your platform is not currently supported you may be able to port it yourself, something like this:
- provide implementations of the functions in the port api; use the existing platform implementations for guidance (e.g. platform/nrf5sdk/src):
- the initialisation and OS interfaces are probably the simplest: you will need task creation/deletion and an entry point into task-land, plain-old non-recursive mutexes, a way to queue things, a way to block a task for x milliseconds, a way to obtain a count of [32-bit] milliseconds since boot and also semaphores,
- the common platform/common/event_queue code will likely form most of your implementation of the u_port_event_queue.h API,
- you will need a way to get debug strings off the platform, i.e. [non-floating point]
printf()
to somewhere, - the GPIO API will require some plumbing into the specifics of your MCU,
- the UART API will likely be the most complex thing to implement; you will probably need to know details of the interrupt and DMA behaviours of your MCU to complete this,
- if you intend to use a GNSS chip connected directly to your MCU via I2C then you will need to port the I2C API,
- if you intend to use a GNSS chip connected directly to your MCU through SPI then you will need to port the SPI API,
- some crypto functions are required if you want to use the security features in
ubxlib
; in our experience these are almost always provided by mbedTLS, in which case no modification to the existing port will be required (excepting differences arising from future mbedTLS versions, e.g. we have not yet integrated with version 3), - if you intend to use the BLE that is on-chip, inside the same host MCU as
ubxlib
is running, you will require an implementation of the GATT access functions, - if your platform does not use newlib (if you are using GCC it will bring newlib with it) then you may find you are missing some C library functions; implementations of C library functions we have already found to be missing on some platforms can be found in port/clib and can just be hooked-in from there but you may need to add more if your code doesn't compile,
- if your platform does not offer
malloc()
andfree()
, or you wish to do your own thing with heap memory, you should override the default, weakly-linked, implementations ofpUPortMalloc()
anduPortFree()
by defining your own implementations of these functions in a file inside thesrc
directory of your port, - if your platform supports setting a time-zone offset you will need to implement
uPortGetTimezoneOffsetSeconds()
; if not then you may simply include the file port/u_port_timezone.c in your build (already included through weak linkage via ubxlib.cmake and ubxlib.mk) to get a default timezone offset of zero.
- provide your own versions of the header files
u_cfg_app_platform_specific.h
,u_cfg_hw_platform_specific.h
,u_cfg_test_platform_specific.h
andu_cfg_os_platform_specific.h
(see examples in the existing platform directories); take particular note of translating the task priority values into those of your OS, - provide your own build metadata files (for CMake, Make, a home-grown Python lash-up, whatever): usually your chosen platform will dictate the shape of these and you just need to add to your existing structure the paths to the
ubxlib
source files and theubxlib
include files; otherwise take a look at the existing nrf5 GCC platform or static_size platforms as a starting point (though note that the latter does not bring in anyplatform
ortest
files), - add Unity to your build and then compile and run the tests in u_port_test.c: if these pass then you have likely completed the necessary porting.
For ports that use CMake ubxlib.cmake can be used to collect the ubxlib
non-platform-specific source code files, include directories etc.
ubxlib.cmake
is typically included in a port specific CMake file and will then define a couple of variables that the calling CMake file can make use of (see the esp32 component CMakeLists.txt as example).
Before including ubxlib.cmake
you must set the variable UBXLIB_BASE
to root directory of ubxlib
.
You must also specify what ubxlib
features to enable using the UBXLIB_FEATURES
variable described below.
To bring the platform-specific files into your CMake file, take a look at, for instance, the Windows-platform CMakeLists.txt file.
Available features currently are:
u_lib
: Include thelib_common
APIshort_range
: Includewifi
,ble
andshort_range
APIcell
: Includecell
APIgnss
: Includegnss
API
set(UBXLIB_BASE <path_to_ubxlib>)
# Set the ubxlib features to compile (all needs to be enabled at the moment)
set(UBXLIB_FEATURES short_range cell gnss)
# From this line we will get back:
# - UBXLIB_SRC
# - UBXLIB_INC
# - UBXLIB_PRIVATE_INC
# - UBXLIB_TEST_SRC
# - UBXLIB_TEST_INC
include(${UBXLIB_BASE}/port/ubxlib.cmake)
After ubxlib.cmake
has been included the following variables will be available:
UBXLIB_SRC
: A list of all the non-platform-specific .c filesUBXLIB_INC
: A list of the public include directoriesUBXLIB_PRIVATE_INC
: A list of the private include directories required to build ubxlibUBXLIB_TEST_SRC
: A list of all the test .c filesUBXLIB_TEST_INC
: A list of test include directories
For ports that use make
ubxlib.mk can be used to collect the non-platform-specific ubxlib
source code files, include directories etc. in the same way as for CMake above.
Just like the shared CMake file you must first set the variable UBXLIB_BASE
to root directory of ubxlib
and configure the ubxlib
features using the UBXLIB_FEATURES
(please see UBXLIB_FEATURES above). ubxlib.mk
will then define the same output variables as the shared CMake above (see Output Variables)
To bring the platform-specific files into your Makefile
, take a look at, for instance, the STM32F4 Makefile.
# Include ubxlib src and inc
UBXLIB_BASE = <path_to_ubxlib>
UBXLIB_FEATURES = cell gnss short_range
# From this line we will get back:
# - UBXLIB_SRC
# - UBXLIB_INC
# - UBXLIB_PRIVATE_INC
# - UBXLIB_TEST_SRC
# - UBXLIB_TEST_INC
include $(UBXLIB_BASE)/port/ubxlib.mk