diff --git a/include/quintain.h b/include/quintain.h index 7179cb2..41a57e2 100644 --- a/include/quintain.h +++ b/include/quintain.h @@ -33,7 +33,12 @@ extern "C" { #define QTN_ERR_UNKNOWN_PROVIDER (-4) /* can't find provider */ /* flags for workload operations */ -#define QTN_WORK_USE_SERVER_POOLSET 1 +enum { + QTN_WORK_USE_SERVER_POOLSET, + QTN_WORK_CACHE_UPDATE, + QTN_WORK_CACHE_WRITE, + QTN_WORK_CACHE_READ +}; #ifdef __cplusplus } diff --git a/src/Makefile.subdir b/src/Makefile.subdir index ca35873..f5eb3ff 100644 --- a/src/Makefile.subdir +++ b/src/Makefile.subdir @@ -5,11 +5,15 @@ src_libquintain_client_la_SOURCES += src/quintain-client.c \ src/quintain-rpc.h src_libquintain_server_la_SOURCES += src/quintain-server.c \ - src/quintain-rpc.h + src/quintain-rpc.h \ + src/hoard.cc \ + src/hoard.hpp \ + src/hoard-c.h dist_bin_SCRIPTS += src/quintain-benchmark-parse.sh if HAVE_MPI bin_PROGRAMS += src/quintain-benchmark src_quintain_benchmark_LDADD = src/libquintain-client.la -lbedrock-client +noinst_HEADERS += src/quintain-macros.h endif diff --git a/src/hoard-c.h b/src/hoard-c.h new file mode 100644 index 0000000..5b1fb98 --- /dev/null +++ b/src/hoard-c.h @@ -0,0 +1,15 @@ +#ifdef __cplusplus +extern "C" { +#endif +#include +#include + +typedef struct Hoard* hoard_t; + +hoard_t hoard_init(); +int hoard_put(hoard_t h, int64_t* src, size_t count, size_t offset); +int hoard_get(hoard_t h, int64_t* dest, size_t count, size_t offset); +void hoard_finalize(hoard_t h); +#ifdef __cplusplus +} +#endif diff --git a/src/hoard.cc b/src/hoard.cc new file mode 100644 index 0000000..c0f4eb5 --- /dev/null +++ b/src/hoard.cc @@ -0,0 +1,17 @@ +#include "hoard.hpp" +#include "hoard-c.h" + +hoard_t hoard_init() +{ + Hoard* h = new (Hoard); + return h; +} +int hoard_put(hoard_t h, int64_t* src, size_t count, size_t offset) +{ + return (h->put(src, count, offset)); +} +int hoard_get(hoard_t h, int64_t* dest, size_t count, size_t offset) +{ + return h->get(dest, count, offset); +} +void hoard_finalize(hoard_t h) { delete h; } diff --git a/src/hoard.hpp b/src/hoard.hpp new file mode 100644 index 0000000..e5b33f2 --- /dev/null +++ b/src/hoard.hpp @@ -0,0 +1,50 @@ +#include +#include +#include +#include + +/* just a big ol' flat array of data. There is no paging out of excess + * data. no least recently used or anything like that. Just how fast + * can we update this data structure concurrently */ + +class Hoard { + public: + Hoard() = default; + int put(int64_t* src, size_t count, size_t offset); + int get(int64_t* dest, size_t count, size_t offset); + + private: + std::vector m_hoard; + void show() + { + for (const auto& x : m_hoard) std::cout << x << " "; + std::cout << std::endl; + } +}; + +int Hoard::put(int64_t* src, size_t count, size_t offset) +{ + if (m_hoard.size() < offset + count) + m_hoard.resize((m_hoard.size() + offset + count) * 2); + + // having trouble using insert() correctly concurrently... + // m_hoard.insert(m_hoard.begin()+offset, src, src+count); + for (size_t i = 0; i < count; i++) m_hoard[offset + i] = src[i]; +#ifdef DEBUG_HOARD + std::cout << "Hoard::put: " << src[0] << "... " << count << " items at " + << offset << std::endl; + ; + show(); +#endif + return count; +} +int Hoard::get(int64_t* dest, size_t count, size_t offset) +{ +#ifdef DEBUG_HOARD + std::cout << "Hoard::get: " << count << " items at " << offset << " " + << m_hoard[offset] << std::endl; + show(); +#endif + for (size_t i = 0; i < count; i++) dest[i] = m_hoard[offset + i]; + return count; +} diff --git a/src/quintain-benchmark.c b/src/quintain-benchmark.c index 1825b73..63baabc 100644 --- a/src/quintain-benchmark.c +++ b/src/quintain-benchmark.c @@ -207,6 +207,9 @@ int main(int argc, char** argv) if (json_object_get_boolean( json_object_object_get(json_cfg, "use_server_poolset"))) work_flags |= QTN_WORK_USE_SERVER_POOLSET; + if (json_object_get_boolean( + json_object_object_get(json_cfg, "concurrency_test"))) + work_flags |= QTN_WORK_CACHE_UPDATE; bulk_size = json_object_get_int(json_object_object_get(json_cfg, "bulk_size")); if (strcmp("pull", json_object_get_string( diff --git a/src/quintain-rpc.h b/src/quintain-rpc.h index f12b674..efdd0d7 100644 --- a/src/quintain-rpc.h +++ b/src/quintain-rpc.h @@ -19,7 +19,11 @@ typedef struct { uint32_t flags; /* flags to modify behavior */ uint32_t bulk_op; /* what type of bulk xfer to do */ hg_bulk_t bulk_handle; /* bulk handle (if set) for bulk xfer */ - char* req_buffer; /* dummy buffer */ + int64_t scratch; /* "cache update" mode: value to store in cache (TODO: use + req_buffer instead) */ + int64_t count; /* "cache update" mode: how many items to update */ + int64_t offset; /* "cache update" mode: where to update */ + char* req_buffer; /* dummy buffer */ } qtn_work_in_t; static inline hg_return_t hg_proc_qtn_work_in_t(hg_proc_t proc, void* v_out_p); diff --git a/src/quintain-server.c b/src/quintain-server.c index 9163096..3cad133 100644 --- a/src/quintain-server.c +++ b/src/quintain-server.c @@ -21,6 +21,9 @@ #include "quintain-rpc.h" #include "quintain-macros.h" +#include "hoard-c.h" + +DECLARE_MARGO_RPC_HANDLER(qtn_get_server_config_ult) DECLARE_MARGO_RPC_HANDLER(qtn_work_ult) static int validate_and_complete_config(struct json_object* _config, @@ -35,6 +38,8 @@ struct quintain_provider { hg_id_t qtn_work_rpc_id; struct json_object* json_cfg; + + hoard_t hoard_id; }; static void quintain_server_finalize_cb(void* data) @@ -130,6 +135,9 @@ int quintain_provider_register(margo_instance_id mid, goto error; } + /* initialize our data strucutre in case we measure concurrency */ + tmp_provider->hoard_id = hoard_init(); + /* register RPCs */ rpc_id = MARGO_REGISTER_PROVIDER(mid, "qtn_work_rpc", qtn_work_in_t, qtn_work_out_t, qtn_work_ult, provider_id, @@ -159,6 +167,7 @@ int quintain_provider_register(margo_instance_id mid, int quintain_provider_deregister(quintain_provider_t provider) { + hoard_finalize(provider->hoard_id); margo_provider_pop_finalize_callback(provider->mid, provider); quintain_server_finalize_cb(provider); return QTN_SUCCESS; @@ -235,6 +244,15 @@ static void qtn_work_ult(hg_handle_t handle) = margo_bulk_transfer(mid, in.bulk_op, info->addr, in.bulk_handle, 0, bulk_handle, 0, in.bulk_size); } + + if (in.flags & QTN_WORK_CACHE_UPDATE) { + if (in.flags & QTN_WORK_CACHE_WRITE) + hoard_put(provider->hoard_id, &(in.scratch), + in.count / sizeof(int64_t), in.offset); + else + hoard_get(provider->hoard_id, &(in.scratch), + in.count / sizeof(int64_t), in.offset); + } if (out.ret != HG_SUCCESS) { QTN_ERROR(mid, "margo_bulk_transfer: %s", HG_Error_to_string(out.ret)); }