fIwJlxSzApHEZIl
fIwJlxSzApHEZIl

Reputation: 13290

Simple Flatbuffers over ZeroMQ C example - Copy struct to flatbuffer over zmq and back to a struct again

Posting my work for posterity. Realized after finishing my last example in C++ that I actually needed to do it in C all along (awesome, right?). Both iterations took me considerable effort as a Java programmer and I think a lot of the sample code out there leaves far too many holes - especially when it comes to building which is considerably more difficult from the command line for someone who is used to using, say Eclipse, to build a project and handle dependencies.

How to install dependencies for OSX with brew:

brew install flatcc
brew install zeromq

You'll need all the standard builder binaries installed as well. I used gcc to compile with:

gcc publisher.c -o bin/zmq_pub -lzmq -lflatcc
gcc subscriber.c -o bin/zmq_sub -lzmq

This assumes you've installed the zmq and flatcc libraries which should get symlinked to your /usr/local/include after brew finishes installing them. Like this:

zmq_cpub $ls -la /usr/local/include lrwxr-xr-x 1 user group 37 Oct 18 18:43 flatcc -> ../Cellar/flatcc/0.3.4/include/flatcc

You'll get compilation errors such as: Undefined symbols for architecture x86_64: if you don't have the libraries correctly installed / linked. The compiler / linker will rename functions and prefix them with _ and potentially confuse the hell out of you. Like Undefined symbols for architecture x86_64 _flatcc_builder_init even though there never is supposed to be an _flatcc_builder_init.

That's because linking libraries in C / C++ is fundamentally different than in Java. Instead of a specific project build path that you add JARs too there are known locations where external C / C++ libraries can install to. /usr/local/include, /usr/local/lib, /usr/lib, and /usr/include.

And don't forget to generate the header files to use in your local project after installing the flatcc binary to your path:

flatcc -a Car.fbs

That should be pretty much every obstacle I faced on my trip down C lane. Hope it helps someone out there.

Upvotes: 3

Views: 2604

Answers (1)

fIwJlxSzApHEZIl
fIwJlxSzApHEZIl

Reputation: 13290

Car.fbs

namespace Test;

table Car {
    name: string;
    model: string;
    year: int;
}
root_type Car;

Subscriber.c (listens for incoming structs)

//  Hello World client
#include "flatbuffers/Car_builder.h" // Generated by `flatcc`.
#include "flatbuffers/flatbuffers_common_builder.h"
#include <zmq.h>

#undef ns
#define ns(x) FLATBUFFERS_WRAP_NAMESPACE(Test, x) // Specified in the schema

struct Car {
    char* name;
    char* model;
    int year;
};

int main (void)
{
    printf ("Connecting to car world server...\n");
    void *context = zmq_ctx_new ();
    void *requester = zmq_socket (context, ZMQ_REQ);
    zmq_connect (requester, "tcp://localhost:5555");

    int request_nbr;
    for (request_nbr = 0; request_nbr != 10; request_nbr++) {
        char buffer [1024];
        printf ("Sending ready signal %d...\n", request_nbr);
        zmq_send (requester, "Hello", 5, 0);
        zmq_recv (requester, buffer, 1024, 0);
        printf ("Received car %d\n", request_nbr);
        ns(Car_table_t) car = ns(Car_as_root(buffer));
        int year = ns(Car_year(car));
        flatbuffers_string_t model = ns(Car_model(car));
        flatbuffers_string_t name = ns(Car_name(car));

        struct Car nextCar;
        // no need to double up on memory!!
        // strcpy(nextCar.model, model);
        // strcpy(nextCar.name, name);
        nextCar.model = model;
        nextCar.name = name;
        nextCar.year = year;

        printf("Year: %d\n", nextCar.year);
        printf("Name: %s\n", nextCar.name);
        printf("Model: %s\n", nextCar.model);
    }
    zmq_close (requester);
    zmq_ctx_destroy (context);
    return 0;
}

Publisher.c (sends structs over zmq socket):

//  Hello World server

#include "flatbuffers/Car_builder.h" // Generated by `flatcc`.
#include "flatbuffers/flatbuffers_common_builder.h"
#include <zmq.h>
#include <unistd.h>
#include <time.h>

#undef ns
#define ns(x) FLATBUFFERS_WRAP_NAMESPACE(Test, x) // specified in the schema

struct Car {
    char name[10];
    char model[10];
    int year;
};

struct Car getRandomCar() {
    struct Car randomCar;
    int a = rand();
    if ((a % 2) == 0) {
        strcpy(randomCar.name, "Ford");
        strcpy(randomCar.model, "Focus");
    } else {
        strcpy(randomCar.name, "Dodge");
        strcpy(randomCar.model, "Charger");
    }
    randomCar.year = rand();
    return randomCar;
}

int main (void)
{
    srand(time(NULL));

    //  Socket to talk to clients
    void *context = zmq_ctx_new ();
    void *responder = zmq_socket (context, ZMQ_REP);
    int rc = zmq_bind (responder, "tcp://*:5555");
    assert (rc == 0);
    int counter = 0;

    while (1) {
        struct Car c = getRandomCar();

        flatcc_builder_t builder, *B;
        B = &builder;
        // Initialize the builder object.
        flatcc_builder_init(B);
        uint8_t *buf; // raw buffer used by flatbuffer
        size_t size; // the size of the flatbuffer
        // Convert the char arrays to strings
        flatbuffers_string_ref_t name = flatbuffers_string_create_str(B, c.name);
        flatbuffers_string_ref_t model = flatbuffers_string_create_str(B, c.model);

        ns(Car_start_as_root(B));
        ns(Car_name_add(B, name));
        ns(Car_model_add(B, model));
        ns(Car_year_add(B, c.year));
        ns(Car_end_as_root(B));
        buf = flatcc_builder_finalize_buffer(B, &size);

        char receiveBuffer [10];
        zmq_recv (responder, receiveBuffer, 10, 0);
        printf ("Received ready signal. Sending car %d.\n", counter);
        sleep (1);          //  Do some 'work'
        zmq_send (responder, buf, size, 0);
        counter++;

        free(buf);
    }
    return 0;
}

Upvotes: 6

Related Questions