/nearobject-framework

Framework for interacting with short-range devices

Primary LanguageC++MIT LicenseMIT

Near Object (NO) Framework

CI/CD cpp-linter

This project is a framework for interacting with short-range devices, providing secure location, ranging, radar, or proximity services. There is a particular focus on IEEE 802.15.4z-2020 Ultra-Wideband (UWB) devices using the Fine Ranging Consortium (FiRa), however, the framework is not limited to this.

Project Structure

This project is organized to allow primary development on both Linux and Windows. Hence, CMake is used as the build system generator. Consequently, there is an OS-independent source tree lib, and OS-dependent source trees windows, linux, etc..

Note that all language feature configuration is constrained by the Windows build system since it is the most limiting factor. As such, the current C++ language version being used is C++ 20.

Coding Guidelines

Where possible, we will attempt to use primitives provided by the C++ Standard Library for interoperability between common and OS-dependent code. The use of OS-specific primitives and libraries is reserved for scenarios where they are strictly needed (eg. calling an OS/System API), or where the highest possible performance is required and only the OS implementation can provide this.

The coding style is dictated by both .clang-format and .clang-tidy files at the root of the project. Please configure your editor to format and lint sources accordingly. Above all, the coding style should be kept as consistent as possible. The exact style used is not overly important.

To help keep the code consistent, please follow these general guidelines:

  • DO use spaces instead of tabs.
  • DON'T prefix variable names to describe their type, scope, or visibility.
  • DO use std::filesystem for storage and UNIX path separators (/) where possible.
  • DO use complete words for variable and function names.
  • DO use the following table for variable naming:
Block Style Example
Types PascalCase struct NearObject {};
Functions PascalCase NearObject GetNearObject()
Variables camelCase NearObject nearObject;
Parameters camelCase void registerEventCallback(NearObjectEventCallback& eventCallback)
Namespaces lowercase namespace nearobject
Public Members PascalCase struct NearObject { NearObjectProperties Properties; }
Private Members camelCase with m_ prefix class NearObjectSession { uint64_t m_sessionId; }

Commit Signing

While not required, it is strongly recommended to configure git with a GNU Privacy Guard (GPG) signing key. This allows GitHub to verify commits were pushed by a specific user and will show a green Verified status beside each verified commit. Follow these steps to configure a signing key for commit verification:

  1. Install the gpg tools for the target operating system (see the OS-specific README files for details).
  2. Generate a gpg key
  3. Add the gpg key to your github account
  4. Configure git to use the gpg key. This link will also tell you how to use gpg signing with an ssh key. Additionally, if you wish tell git to use a type any type of signing by default, be it gpg, ssh or X.509, you need to run the following command git config --global commit.gpgsign true

Development Environment Setup

Pre-requisites:

  • C++ 20 Compiler
  • CMake version >= 3.16

Compiler

Both a compiler and standard C++ library supporting C++20 are required. The C++ reference pages for C++20 core language features and C++20 library features provide complete details about current compiler and library support.

Windows

Visual Studio 2022 generally satisfies the requirements, however, the full integrated development environment (IDE) is not needed unless also building the simulator driver. A much leaner alternative for those using other editors such as Visual Studio Code can instead install Visual Studio Build Tools. The build tools come with a C++ compatible compiler and standard library. Detailed development environment setup instructions can be found in the Windows README.

Linux

g++ or llvm/clang are suitable, however, some care must be taken to obtain a compatible standard library. A known, working environment is ubuntu 22.04 (jammy) with clang 14.0.0 and LLVM 14.0.0. Both are both provided by the official ubuntu package repository so can be installed using apt. Detailed development environment setup instructions can be found in the Linux README.

CMake

CMake may be installed in any form, as long as the version meets the minimum. One popular way of installing it on Windows is to use Chocolately with choco install -y cmake. On Linux, all standard package managers provide a cmake package (eg. apt-get install -y cmake, yum install -y cmake, etc.).

To bootstrap the build environment, instruct CMake to generate the build files. It is strongly recommended to do this in a directory that is separate from the source; this allows one to easily destroy and recreate the build environment without affecting the checked-out source and changes in progress. Typically, a new directory called build at the top-level project tree is used for this purpose:

git clone git@github.com:microsoft/nearobject-framework.git
cd nearobject-framework
cmake -DCMAKE_EXPORT_COMPILE_COMMANDS:BOOL=TRUE -Bbuild 
cmake --build build

CMake with Visual Studio Code

Alternatively, Microsoft provides a CMake Tools Visual Studio Code extension that automates this process. After installing the extension and cloning the repository in VSCode, hit Ctrl+Shift+P, then find the command CMake: Delete Cache and Reconfigure. This will generate the build configuration in a build folder at the top-level of the source tree. Once done, you can build the ALL target (default) with the CMake: Build command again, Ctrl+Shift+P, type cmake, find the command).

In general, you set a build target and variant, then use the CMake: Build command to build incrementally. All build targets can be found in the CMake: Project Outline activity bar, but a list of them will also be shown when invoking actions that involve targets:

Source Dependencies

Source dependencies can be satisfied in two (2) ways:

  1. From a URL using the CMake FetchContent module (default).
  2. From the vcpkg package manager.

The FetchContent method works out of the box with no setup required, so is the default method. The vcpkg method requires one-time bootstrapping, described below.

vcpkg

The vcpkg source dependency method is configurable through the CMake build option NOF_USE_VCPKG. This option defaults to OFF since FetchContent is much faster for development loop tasks. Set this to ON to use vcpkg for source dependency resolution (eg. via command line argument such as cmake -DNOF_USE_VCPKG:BOOL=ON ., or by editing the CMake cache directly). Note that this is a cache setting, thus it is sticky; once set, the chosen setting will be stored in the CMake variable cache (CMakeCache.txt) and so it must be explicitly reset to OFF when no longer desired.

vcpkg has extra dependencies on Linux including curl, zip, unzip, tar, and pkg-config. These can be installed on Debian or Ubuntu distributions using the following command:

sudo apt-get install curl zip unzip tar pkg-config

Once configured in the build, the vcpkg binary must be made available. This can be installed globally for the entire system, locally for the current user, or directly in the project workspace.

For project-direct installation, the project has been configured to include vcpkg source as a sub-module @ ./vcpkg and attempt to automatically bootstrap it. If this fails for some reason, the following steps can be taken to manually bootstrap vcpkg:

The vcpkg submodule must first be initialized and updated to sync it to the fixed, known working release tag, which is accomplished by issuing the following git command at the project root:

git submodule update --init --recursive

Next, run the OS-specific bootstapping script (./vcpkg/bootstrap-vcpkg.[bat|sh]). This should produce the binary ./vcpkg/vcpkg.