The Azure SDK for Embedded C is designed to allow small embedded (IoT) devices to communicate with Azure services. Since we expect our client library code to run on microcontrollers, which have very limited amounts of flash and RAM, and have slower CPUs, our C SDK does things very differently than the SDKs we offer for other languages.
With this in mind, there are many tenants or principles that we follow in order to properly address this target audience:
-
Customers of our SDK compile our source code along with their own.
-
We target the C99 programming language and test with gcc, clang, & MS Visual C compilers.
-
We offer very few abstractions making our code easy to understand and debug.
-
Our SDK is non allocating. That is, customers must allocate our data structures where they desire (global memory, heap, stack, etc.) and then pass the address of the allocated structure into our functions to initialize them and in order to perform various operations.
-
Unlike our other language SDKs, many things (such as composing an HTTP pipeline of policies) are done in source code as opposed to runtime. This reduces code size, improves execution speed and locks-in behavior, reducing the chance of bugs at runtime.
-
We support microcontrollers with no operating system, microcontrollers with a real-time operating system (like Azure RTOS), Linux, and Windows. Customers can implement their own "platform layer" to use our SDK on devices we don’t support out-of-the-box. The platform layer requires minimal functionality such as a clock, a mutex, sleep, and an HTTP stack. We provide some platform layers, and more will be added over time.
To get help with the SDK:
- File a Github Issue.
- Ask new questions or see others' questions on Stack Overflow using the
azure
andc
tags.
The master branch has the most recent code with new features and bug fixes. It does not represent the latest General Availability (GA) release of the SDK.
When we make an official release, we will create a unique git tag containing the name and version to mark the commit. We'll use this tag for servicing via hotfix branches as well as debugging the code for a particular preview or stable release version. A release tag looks like this:
<package-name>_<package-version>
For more information, please see this branching strategy document.
-
Install the required prerequisites:
-
Clone our Azure SDK repository, optionally using the desired version tag.
git clone https://github.com/Azure/azure-sdk-for-c git checkout <tag_name>
For information about using a specific client library, see the README file located in the client library's folder which is a subdirectory under the
/sdk
folder. -
Ensure the SDK builds correctly.
-
Create an output directory for your build artifacts (in this example, we named it
build
, but you can pick any name).mkdir build
-
Navigate to that newly created directory.
cd build
-
Run
cmake
pointing to the sources at the root of the repo to generate the builds files.cmake ..
-
Launch the underlying build system to compile the libraries.
cmake --build .
This results in building each library as a static library file, placed in the output directory you created (for example
build\sdk\core\core\Debug
). At a minimum, you must have anAzure Core
library, aPlatform
library, and anHTTP
library. Then, you can build any additional Azure service client library you intend to use from within your application (for examplebuild\sdk\storage\blobs\Debug
). To use our client libraries in your application, just#include
our public header files and then link your application's object files with our library files. -
-
Provide platform-specific implementations for functionality required by
Azure Core
. For more information, see the Azure Core Porting Guide.
By default, when building the project with no options, the following static libraries are generated:
Libraries
:- az_core
- az_span, az_http, az_json, etc.
- az_iot
- iot_provisioning, iot_hub, etc.
- az_storage_blobs
- Storage SDK blobs client.
- az_noplatform
- Library that provides a basic returning error for platform abstraction as AZ_NOT_IMPLEMENTED. This ensures the project can be compiled without the need to provide any specific platform implementation. This is useful if you want to use az_core without platform specific functions like
mutex
ortime
.
- Library that provides a basic returning error for platform abstraction as AZ_NOT_IMPLEMENTED. This ensures the project can be compiled without the need to provide any specific platform implementation. This is useful if you want to use az_core without platform specific functions like
- az_nohttp
- Library that provides a basic returning error when calling HTTP stack. Similar to az_noplatform, this library ensures the project can be compiled without requiring any HTTP stack implementation. This is useful if you want to use
az_core
withoutaz_http
functionality.
- Library that provides a basic returning error when calling HTTP stack. Similar to az_noplatform, this library ensures the project can be compiled without requiring any HTTP stack implementation. This is useful if you want to use
- az_core
The following compiler options are available for adding/removing project features.
Option | Description | Default Value |
UNIT_TESTING | Generates Unit Test for compilation. When turning this option ON, cmocka is a required dependency for compilation. After Compiling, use `ctest` to run Unit Test. |
OFF |
UNIT_TESTING_MOCKS | This option works only with GCC. It uses -ld option from linker to mock functions during unit test. This is used to test platform or HTTP functions by mocking the return values. | OFF |
PRECONDITIONS | Turning this option ON would remove all method contracts. This us typically for shipping libraries for production to make it as much optimized as possible. | ON |
TRANSPORT_CURL | This option requires Libcurl dependency to be available. It generates an HTTP stack with libcurl for az_http to be able to send requests thru the wire. This library would replace the no_http. | OFF |
TRANSPORT_PAHO | This option requires paho-mqtt dependency to be available. Provides Paho MQTT support for iot. | OFF |
AZ_PLATFORM_IMPL | This option can be set to any of the next values: - No_value: default value is used and no_platform library is used. - "POSIX": Provides implementation for Linux and Mac systems. - "WIN32": Provides platform implementation for Windows based system - "USER": Tells cmake to use an specific implementation provided by user. When setting this option, user must provide an implementation library and set option `AZ_USER_PLATFORM_IMPL_NAME` with the name of the library (i.e. -DAZ_PLATFORM_IMPL=USER -DAZ_USER_PLATFORM_IMPL_NAME=user_platform_lib ). cmake will look for this library to link az_core |
No_value |
-
Samples
: Whenever UNIT_TESTING is ON, samples are built using the default PAL (see running samples section). This means that running samples would throw errors like:./keys_client_example Running sample with no_op HTTP implementation. Recompile az_core with an HTTP client implementation like CURL to see sample sending network requests. i.e. cmake -DTRANSPORT_CURL=ON ..
See compiler options section to learn about how to build samples with HTTP implementation in order to be runnable.
After building samples with HTTP stack, set the environment variables for credentials. The samples read these environment values to authenticate to Azure services. See client secret here for additional details on Azure authentication.
# On linux, set env var like this. For Windows, do it from advanced settings/ env variables
# KEY-VAULT Sample
export AZURE_TENANT_ID="????????-????-????-????-????????????"
export AZURE_CLIENT_ID="????????-????-????-????-????????????"
export AZURE_CLIENT_SECRET="????????????"
export AZURE_KEYVAULT_URL="https://???????????.??"
# STORAGE Sample (only 1 env var required)
# URL must contain a valid container, blob and SaS token
# e.g "https://storageAccount.blob.core.windows.net/container/blob?sv=xxx&ss=xx&srt=xx&sp=xx&se=xx&st=xxx&spr=https,http&sig=xxx"
export AZURE_STORAGE_URL="https://??????????????"
Project contains files to work on Windows, Mac or Linux based OS.
Note For any environment variables set to use with CMake, the environment variables must be set
BEFORE the first cmake generation command (cmake ..
). The environment variables will NOT be picked up
if you have already generated the build files, set environment variables, and then regenerate. In that
case, you must either delete the CMakeCache.txt
file or delete the folder in which you are generating build
files and start again.
vcpkg is the easiest way to have dependencies installed. It downloads packages sources, headers and build libraries for whatever TRIPLET is set up (platform/arq). VCPKG maintains any installed package inside its own folder, allowing to have multiple vcpkg folder with different dependencies installed on each. This is also great because you don't have to install dependencies globally on your system.
Follow next steps to install VCPKG and have it linked to cmake
# Clone vcpkg:
git clone https://github.com/Microsoft/vcpkg.git
# (consider this path as PATH_TO_VCPKG)
cd vcpkg
# build vcpkg (remove .bat on Linux/Mac)
.\bootstrap-vcpkg.bat
# install dependencies (remove .exe in Linux/Mac) and update triplet
vcpkg.exe install --triplet x64-windows-static curl[winssl] cmocka paho-mqtt
# Add this environment variables to link this VCPKG folder with cmake:
# VCPKG_DEFAULT_TRIPLET=x64-windows-static
# VCPKG_ROOT=PATH_TO_VCPKG (replace PATH_TO_VCPKG for where vcpkg is installed)
Follow next steps to build project from command prompt:
# cd to project folder
cd azure_sdk_for_c
# create a new folder to generate cmake files for building (i.e. build)
mkdir build
cd build
# generate files
# cmake will automatically detect what C compiler is used by system by default and will generate files for it
cmake ..
# compile files. Cmake would call compiler and linker to generate libs
cmake --build .
Note: The steps above would compile and generate the default output for azure-sdk-for-c which includes static libraries only. See section Compiler Options
Open project folder with Visual Studio. If VCPKG has been previously installed and set up like mentioned above. Everything will be ready to build. Right after opening project, Visual Studio will read cmake files and generate cache files automatically.
VCPKG can be used to download packages sources, headers and build libraries for whatever TRIPLET is set up (platform/architecture). VCPKG maintains any installed package inside its own folder, allowing to have multiple vcpkg folder with different dependencies installed on each. This is also great because you don't have to install dependencies globally on your system.
Follow next steps to install VCPKG and have it linked to cmake
# Clone vcpkg:
# (consider this path as PATH_TO_VCPKG)
cd vcpkg
# build vcpkg
./bootstrap-vcpkg.sh
./vcpkg install --triplet x64-linux curl cmocka paho-mqtt
export VCPKG_DEFAULT_TRIPLET=x64-linux
export VCPKG_ROOT=PATH_TO_VCPKG #replace PATH_TO_VCPKG for where vcpkg is installed
Alternatively, for Ubuntu 18.04 you can use:
sudo apt install build-essential cmake libcmocka-dev libcmocka0 gcovr lcov doxygen curl libcurl4-openssl-dev libssl-dev ca-certificates
# cd to project folder
cd azure_sdk_for_c
# create a new folder to generate cmake files for building (i.e. build)
mkdir build
cd build
# generate files
# cmake will automatically detect what C compiler is used by system by default and will generate files for it
cmake ..
# compile files. Cmake would call compiler and linker to generate libs
make
Note: The steps above would compile and generate the default output for azure-sdk-for-c which includes static libraries only. See section Compiler Options
VCPKG can be used to download packages sources, headers and build libraries for whatever TRIPLET is set up (platform/architecture). VCPKG maintains any installed package inside its own folder, allowing to have multiple vcpkg folder with different dependencies installed on each. This is also great because you don't have to install dependencies globally on your system.
First, ensure that you have the latest gcc
installed:
brew update
brew upgrade
brew info gcc
brew install gcc
brew cleanup
Follow next steps to install VCPKG and have it linked to cmake
# Clone vcpkg:
git clone https://github.com/Microsoft/vcpkg.git
# (consider this path as PATH_TO_VCPKG)
cd vcpkg
# build vcpkg
./bootstrap-vcpkg.sh
./vcpkg install --triplet x64-osx curl cmocka paho-mqtt
export VCPKG_DEFAULT_TRIPLET=x64-osx
export VCPKG_ROOT=PATH_TO_VCPKG #replace PATH_TO_VCPKG for where vcpkg is installed
# cd to project folder
cd azure_sdk_for_c
# create a new folder to generate cmake files for building (i.e. build)
mkdir build
cd build
# generate files
# cmake will automatically detect what C compiler is used by system by default and will generate files for it
cmake ..
# compile files. Cmake would call compiler and linker to generate libs
make
Note: The steps above would compile and generate the default output for azure-sdk-for-c which includes static libraries only. See section Compiler Options
At the heart of our SDK is, what we refer to as, Azure Core. This code defines several data types and functions for use by the client libraries that build on top of us such as an Azure Storage Blob client library and Azure IoT client libraries. Here are some of the features that customers use directly:
-
Spans: A span represents a byte buffer and is used for string manipulations, HTTP requests/responses, building/parsing JSON payloads. It allows us to return a substring within a larger string without any memory allocations. See the Working With Spans section of the
Azure Core
README for more information. -
Logging: As our SDK performs operations, it can send log messages to a customer-defined callback. Customers can enable this to assist with debugging and diagnosing issues when leveraging our SDK code. See the Logging SDK Operations section of the
Azure Core
README for more information. -
Contexts: Contexts offer an I/O cancellation mechanism. Multiple contexts can be composed together in your application’s call tree. When a context is canceled, its children are also canceled. See the Canceling an Operation section of the
Azure Core
README for more information. -
JSON: Non-allocating JSON builder and JSON parsing data structures and operations.
-
HTTP: Non-allocating HTTP request and HTTP response data structures and operations.
-
Argument Validation: The SDK validates function arguments and invokes a callback when validation fails. By default, this callback suspends the calling thread forever. However, you can override this behavior and, in fact, you can disable all argument validation to get smaller and faster code. See the SDK Function Argument Validation section of the
Azure Core
README for more information.
In addition to the above features, Azure Core
provides features available to client libraries written to access other Azure services. Customers use these features indirectly by way of interacting with a client library. By providing these features in Azure Core
, the client libraries built on top of us will share a common implementation and many features will behave identically across client libraries. For example, Azure Core
offers a standard set of credential types and an HTTP pipeline with logging, retry, and telemetry policies.
For details on contributing to this repository, see the contributing guide.
This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.microsoft.com.
When you submit a pull request, a CLA-bot will automatically determine whether you need to provide a CLA and decorate the PR appropriately (e.g., label, comment). Simply follow the instructions provided by the bot. You will only need to do this once across all repositories using our CLA.
This project has adopted the Microsoft Open Source Code of Conduct. For more information see the Code of Conduct FAQ or contact [email protected] with any additional questions or comments.
Many people all over the world have helped make this project better. You'll want to check out:
- What are some good first issues for new contributors to the repo?
- How to build and test your change
- How you can make a change happen!
Security issues and bugs should be reported privately, via email, to the Microsoft Security Response Center (MSRC) [email protected]. You should receive a response within 24 hours. If for some reason you do not, please follow up via email to ensure we received your original message. Further information, including the MSRC PGP key, can be found in the Security TechCenter.
Azure SDK for Embedded C is licensed under the MIT license.