Skip to content

Latest commit

 

History

History
239 lines (181 loc) · 8.62 KB

README.md

File metadata and controls

239 lines (181 loc) · 8.62 KB

hiPPo

hiPPo is a C++ client designed to communicate with SoHal (HP Sprout Immersive Computer's Hardware Abstraction Layer). The devices and methods in hiPPo mirror the SoHal spec, but hiPPo handles all of the websocket communication and JSONRPC-2.0 protocol for the user.

Note: Please note that hiPPo is currently in heavy development stage and there may be a few bugs in it. Please report any bugs and they will be promptly fixed

Currently the hiPPo API has not yet implemented the following functions that are in the documentation found in the 5_24_2018 version of SoHAL:

  • system.temperatures with a parameter to specifiy which temperatures to get (system.temperatures without parameters is implemented)
  • depthcamera.enable_filter()

As an example, the following code should turn on Sprout's projector:

#include "include/projector.h"

int main(int argc, char *argv[]) {
  uint64_t err = 0;
  uint32_t open_count = 0;
  hippo::Projector projector;

  if (err = projector.open(&open_count)) {
    // handle the error
  }
  if (err = projector.on()) {
    // handle the error
  }
  if (err = projector.close()) {
    // handle the error
  }
  return 0;
}

Implementation details

Standalone library

Currently hiPPo is a standalone library, as all hiPPo dependencies have been linked statically into hippo.dll.

hiPPo uses the libwebsockets project (https://libwebsockets.org) for its websockets communication with SoHal.

hiPPo uses the nlohmann's json project (https://github.com/nlohmann/json) for generating and parsing the JSONRPC messages sent to SoHal.

Overloaded class methods

While we're trying to verbatim copy SoHal's API, C++ needs a bit more 'fillcode'. For example, in hippy (Python sistership client), in order to open the projector we need the following code (assuming the object is already created):

open_count = projector.open()

but, of course, we can also just 'forget' about the open_count return value (containing the number of clients that currently hold the projector open) if we just want to open it:

projector.open()

In order to enable a similar functionality in C++, we're overloading the class methods. For example, the open function has two different prototypes:

  virtual uint64_t open();
  virtual uint64_t open(uint32_t *open_count);

so, the C++ code can be written as:

  if (err = projector.open()) {
    // handle the error
  }

or

  uint32_t open_count = 0;
  if (err = projector.open(&open_count)) {
    // handle the error
  }

This is a bit more interesting in functions like projector.keystone, where the set variant of the method also returns the actual values it has been set to. In this case we have a (hippo::Keystone *get) prototype for getting the keystone value, a (const hippo::Keystone &set, hippo::Keystone *get) to set the keystone and get the actual value it has been set to (in some cases it may differ) back from SoHal, and a (const hippo::Keystone &set) prototype for setting the keystone but not needed to receive the actual set value.

  uint64_t keystone(hippo::Keystone *get);
  uint64_t keystone(const hippo::Keystone &set);
  uint64_t keystone(const hippo::Keystone &set, hippo::Keystone *get);

This allow us to skip the parsing of the return value back from SoHal's into C++ if the user does not needed it. You will notice that all the functions follow this approach when possible.

Note: Please note that we recommend to always check the return value for the projector.keystone function, but we expose these functions for consistency with the rest of the API.

Software Devices

Recent versions of SoHal allow specifically written pieces of software to register themselves on SoHal as a Software Device. These devices inform SoHal as to which functions they implement, and allow a SoHal client to connect to the software device and call those functions. Because the data is passed between SoHal and the software device via JSONRPC, the data must be serializable. An example of a software device (adder) that implements a few sample functions can be found in the swdevices subfolder. The functions that the sample device implements is defined in the adder.json file within that subfolder. Function parameters that can be modeled as c-style structs can also be passed between a software device client and server by defining the data type as a json schema. Examples of these schemas can be found in the swdevices\schemas subfolder.

A complete example adder software device is located in the swdevices subfolder. To compile the adder software device, first compile the hippo library by opening & compiling the win\hippo.sln project. Then open a console window and cd to the swdevices subfolder and type:

python gen_cpp.py --json adder.json --schemas schemas

This will run a python script that creates two files:

  1. swdevices\src\adder.cc
  2. swdevices\include\adder.h

After these two files are created, open the `swdevces\hippo_swdevices.sln" solution and compile both the client and server projects. This will create a software device server that can connect to and register itelf with SoHal, and a client that will call into the server to perform various tasks.

To use the software device the user should perform the following steps in order:

  1. Launch SoHal
  2. Launch the compiled swdevices_server.exe
  3. Launch the compiled swdevices_client.exe

A more complete client/server adder software device can be found in the test\test_swdevice.cc file. In this case a Blackadder server class registers on SoHal and an adder client connects to the Blackadder server to call even more functions than the example shown in the swdevices subfolder.

Notifications

hiPPo now allows Software Device servers to send notifications to clients. In order to do so, the software device server can call the SendNotification function. Currently the notification sends a notification name, and an optional parameter as part of the notification. The data types that the parameter can contain is limited to the following types:

  1. int
  2. float
  3. bool
  4. c-style strings passed in via null terminated char*
  5. wcharptr type
  6. b64bytes type
  7. no parameter

Internally, the server will add an <SoftwareDeviceName>.on_ to the notification name, i.e. calling

SendNotification("slow_call","tick")

from an adder software device server will result in the client receiving a notification

adder.on_slow_call with a SWDeviceNotificationParam parameter that contains tick in the charData field.

In order for the software device client to receive notifications it must be subscribed to the notifications. This can be achieved via a call to subscribe. The client must know the type of parameter for each notification, as a SWDeviceNotificationParam will load as many of its fields with the parameter data as is possible.

An example of the Blackadder software device server sending notifications every 1 second to the client can be found in the test\test_swdevice.cc file.

No exceptions thrown

hiPPo error management is based in old-style functions returning error codes, so, no exceptions will be thrown from hiPPo.

Error codes generated in SoHal and sent back to hiPPo or generated in hiPPo itself, are returned to the user as a uint64_t value. HiPPo also provides the hippo::strerror() API that will return a human-readable string representation of the last error occurred.

Please note that hiPPo error codes follow the same structure than the original SoHal error codes: in the 64 bit error code we include a hash of the original source .cc file name and line number where error was generated together with the actual error code. This has proven very helpful when debugging.

Source code style guide

We use Google's style guide for hiPPo. This was not our original preferred style, but somebody wrote a very clear and useful description in https://google.github.io/styleguide/cppguide.html and they also have cpplint, a tool to automatically check your code. This is very cool and totally worth it to use. And we're already getting used to it ;)

So yes, we're running hiPPo code through cpplint (for C++) before accepting a pull request. We are also setting the warning level to the maximum and treating all warnings as errors. Please make sure you follow this guide if you submit any pull request. Thank you!

Installation

In order to compile hiPPo you will need a folder with its dependencies. Please contact any of the developers to get access to it if needed.

HiPPo does not depend on any library at run-time, except the ones from the operating system and Visual Studio runtime.