2016-07-08 14:44:23 -04:00
|
|
|
/*
|
2015-12-08 15:45:58 -07:00
|
|
|
*------------------------------------------------------------------
|
|
|
|
* persist.c - persistent data structure storage test / demo code
|
|
|
|
*
|
|
|
|
* Copyright (c) 2013 Cisco and/or its affiliates.
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at:
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <vppinfra/clib.h>
|
|
|
|
#include <vppinfra/vec.h>
|
|
|
|
#include <vppinfra/hash.h>
|
|
|
|
#include <vppinfra/bitmap.h>
|
|
|
|
#include <vppinfra/fifo.h>
|
|
|
|
#include <vppinfra/time.h>
|
|
|
|
#include <vppinfra/mheap.h>
|
|
|
|
#include <vppinfra/heap.h>
|
|
|
|
#include <vppinfra/pool.h>
|
|
|
|
#include <vppinfra/format.h>
|
|
|
|
#include <vppinfra/serialize.h>
|
|
|
|
#include <svmdb.h>
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
svmdb_client_t *c;
|
2015-12-08 15:45:58 -07:00
|
|
|
} persist_main_t;
|
|
|
|
|
|
|
|
persist_main_t persist_main;
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
u8 *string1;
|
|
|
|
u8 *string2;
|
2015-12-08 15:45:58 -07:00
|
|
|
} demo_struct2_t;
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
demo_struct2_t *demo2;
|
|
|
|
u8 *name;
|
2015-12-08 15:45:58 -07:00
|
|
|
} demo_struct1_t;
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
/*
|
2015-12-08 15:45:58 -07:00
|
|
|
* Data structures in persistent shared memory, all the time
|
|
|
|
*/
|
2016-07-08 14:44:23 -04:00
|
|
|
clib_error_t *
|
|
|
|
persist_malloc (persist_main_t * pm)
|
2015-12-08 15:45:58 -07:00
|
|
|
{
|
2016-07-08 14:44:23 -04:00
|
|
|
demo_struct2_t *demo2;
|
|
|
|
demo_struct1_t *demo1;
|
|
|
|
time_t starttime = time (0);
|
|
|
|
char *datestring = ctime (&starttime);
|
|
|
|
void *oldheap;
|
|
|
|
|
|
|
|
/* Get back the root pointer */
|
|
|
|
demo1 = svmdb_local_get_variable_reference
|
|
|
|
(pm->c, SVMDB_NAMESPACE_VEC, "demo1_location");
|
|
|
|
|
|
|
|
/* It doesnt exist create our data structures */
|
|
|
|
if (demo1 == 0)
|
|
|
|
{
|
|
|
|
/* If you want MP / thread safety, lock the region... */
|
|
|
|
pthread_mutex_lock (&pm->c->db_rp->mutex);
|
|
|
|
|
|
|
|
/* Switch to the shared memory region heap */
|
|
|
|
oldheap = svm_push_data_heap (pm->c->db_rp);
|
|
|
|
|
|
|
|
/* Allocate the top-level structure as a single element vector */
|
|
|
|
vec_validate (demo1, 0);
|
|
|
|
|
|
|
|
/* Allocate the next-level structure as a plain old memory obj */
|
|
|
|
demo2 = clib_mem_alloc (sizeof (*demo2));
|
|
|
|
|
|
|
|
demo1->demo2 = demo2;
|
|
|
|
demo1->name = format (0, "My name is Ishmael%c", 0);
|
|
|
|
demo2->string1 = format (0, "Here is string1%c", 0);
|
|
|
|
demo2->string2 = format (0, "Born at %s%c", datestring, 0);
|
|
|
|
|
|
|
|
/* Back to the process-private heap */
|
|
|
|
svm_pop_heap (oldheap);
|
|
|
|
pthread_mutex_unlock (&pm->c->db_rp->mutex);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the root pointer. Note: this guy switches heaps, locks, etc.
|
|
|
|
* We allocated demo1 as a vector to make this "just work..."
|
|
|
|
*/
|
|
|
|
svmdb_local_set_vec_variable (pm->c, "demo1_location",
|
|
|
|
demo1, sizeof (demo1));
|
2015-12-08 15:45:58 -07:00
|
|
|
|
|
|
|
}
|
2016-07-08 14:44:23 -04:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* retrieve and print data from shared memory */
|
|
|
|
demo2 = demo1->demo2;
|
|
|
|
fformat (stdout, "name: %s\n", demo1->name);
|
|
|
|
fformat (stdout, "demo2 location: %llx\n", demo2);
|
|
|
|
fformat (stdout, "string1: %s\n", demo2->string1);
|
|
|
|
fformat (stdout, "string2: %s\n", demo2->string2);
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
2016-07-08 14:44:23 -04:00
|
|
|
return 0;
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
void
|
|
|
|
unserialize_demo1 (serialize_main_t * sm, va_list * args)
|
2015-12-08 15:45:58 -07:00
|
|
|
{
|
2016-07-08 14:44:23 -04:00
|
|
|
demo_struct1_t **result = va_arg (*args, demo_struct1_t **);
|
|
|
|
demo_struct1_t *demo1;
|
|
|
|
demo_struct2_t *demo2;
|
|
|
|
|
|
|
|
/* Allocate data structures in process private memory */
|
|
|
|
demo1 = clib_mem_alloc (sizeof (*demo1));
|
|
|
|
demo2 = clib_mem_alloc (sizeof (*demo2));
|
|
|
|
demo1->demo2 = demo2;
|
|
|
|
|
|
|
|
/* retrieve data from shared memory checkpoint */
|
|
|
|
unserialize_cstring (sm, (char **) &demo1->name);
|
|
|
|
unserialize_cstring (sm, (char **) &demo2->string1);
|
|
|
|
unserialize_cstring (sm, (char **) &demo2->string2);
|
|
|
|
*result = demo1;
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
void
|
|
|
|
serialize_demo1 (serialize_main_t * sm, va_list * args)
|
2015-12-08 15:45:58 -07:00
|
|
|
{
|
2016-07-08 14:44:23 -04:00
|
|
|
demo_struct1_t *demo1 = va_arg (*args, demo_struct1_t *);
|
|
|
|
demo_struct2_t *demo2 = demo1->demo2;
|
2015-12-08 15:45:58 -07:00
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
serialize_cstring (sm, (char *) demo1->name);
|
|
|
|
serialize_cstring (sm, (char *) demo2->string1);
|
|
|
|
serialize_cstring (sm, (char *) demo2->string2);
|
|
|
|
}
|
2015-12-08 15:45:58 -07:00
|
|
|
|
|
|
|
/* Serialize / unserialize variant */
|
2016-07-08 14:44:23 -04:00
|
|
|
clib_error_t *
|
2015-12-08 15:45:58 -07:00
|
|
|
persist_serialize (persist_main_t * pm)
|
|
|
|
{
|
2016-07-08 14:44:23 -04:00
|
|
|
u8 *checkpoint;
|
|
|
|
serialize_main_t sm;
|
|
|
|
|
|
|
|
demo_struct2_t *demo2;
|
|
|
|
demo_struct1_t *demo1;
|
|
|
|
time_t starttime = time (0);
|
|
|
|
char *datestring = ctime (&starttime);
|
|
|
|
|
|
|
|
/* Get back the root pointer */
|
|
|
|
checkpoint = svmdb_local_get_vec_variable (pm->c, "demo1_checkpoint",
|
|
|
|
sizeof (u8));
|
|
|
|
|
|
|
|
/* It doesnt exist create our data structures */
|
|
|
|
if (checkpoint == 0)
|
|
|
|
{
|
|
|
|
/* Allocate data structures in process-private memory */
|
|
|
|
demo1 = clib_mem_alloc (sizeof (*demo2));
|
|
|
|
vec_validate (demo1, 0);
|
|
|
|
demo2 = clib_mem_alloc (sizeof (*demo2));
|
|
|
|
|
|
|
|
demo1->demo2 = demo2;
|
|
|
|
demo1->name = format (0, "My name is Ishmael%c", 0);
|
|
|
|
demo2->string1 = format (0, "Here is string1%c", 0);
|
|
|
|
demo2->string2 = format (0, "Born at %s%c", datestring, 0);
|
|
|
|
|
|
|
|
/* Create checkpoint */
|
|
|
|
serialize_open_vector (&sm, checkpoint);
|
|
|
|
serialize (&sm, serialize_demo1, demo1);
|
|
|
|
checkpoint = serialize_close_vector (&sm);
|
|
|
|
|
|
|
|
/* Copy checkpoint into shared memory */
|
|
|
|
svmdb_local_set_vec_variable (pm->c, "demo1_checkpoint",
|
|
|
|
checkpoint, sizeof (u8));
|
|
|
|
/* Toss the process-private-memory original.. */
|
|
|
|
vec_free (checkpoint);
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
2016-07-08 14:44:23 -04:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Open the checkpoint */
|
|
|
|
unserialize_open_data (&sm, checkpoint, vec_len (checkpoint));
|
|
|
|
unserialize (&sm, unserialize_demo1, &demo1);
|
|
|
|
|
|
|
|
/* Toss the process-private-memory checkpoint copy */
|
|
|
|
vec_free (checkpoint);
|
|
|
|
|
|
|
|
/* Off we go... */
|
|
|
|
demo2 = demo1->demo2;
|
|
|
|
fformat (stdout, "name: %s\n", demo1->name);
|
|
|
|
fformat (stdout, "demo2 location: %llx\n", demo2);
|
|
|
|
fformat (stdout, "string1: %s\n", demo2->string1);
|
|
|
|
fformat (stdout, "string2: %s\n", demo2->string2);
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
2016-07-08 14:44:23 -04:00
|
|
|
return 0;
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
int
|
|
|
|
main (int argc, char **argv)
|
2015-12-08 15:45:58 -07:00
|
|
|
{
|
2016-07-08 14:44:23 -04:00
|
|
|
unformat_input_t _input, *input = &_input;
|
|
|
|
persist_main_t *pm = &persist_main;
|
|
|
|
clib_error_t *error = 0;
|
|
|
|
|
|
|
|
/* Make a 4mb database arena, chroot so it's truly private */
|
|
|
|
pm->c = svmdb_map_chroot_size ("/ptest", 4 << 20);
|
|
|
|
|
|
|
|
ASSERT (pm->c);
|
|
|
|
|
|
|
|
unformat_init_command_line (input, argv);
|
|
|
|
|
|
|
|
while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
|
|
|
|
{
|
|
|
|
if (unformat (input, "malloc"))
|
|
|
|
error = persist_malloc (pm);
|
|
|
|
else if (unformat (input, "serialize"))
|
|
|
|
error = persist_serialize (pm);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
error = clib_error_return (0, "Unknown flavor '%U'",
|
|
|
|
format_unformat_error, input);
|
|
|
|
break;
|
|
|
|
}
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
svmdb_unmap (pm->c);
|
2015-12-08 15:45:58 -07:00
|
|
|
|
2016-07-08 14:44:23 -04:00
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
clib_error_report (error);
|
|
|
|
exit (1);
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
2016-07-08 14:44:23 -04:00
|
|
|
return 0;
|
2015-12-08 15:45:58 -07:00
|
|
|
}
|
2016-07-08 14:44:23 -04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* fd.io coding-style-patch-verification: ON
|
|
|
|
*
|
|
|
|
* Local Variables:
|
|
|
|
* eval: (c-set-style "gnu")
|
|
|
|
* End:
|
|
|
|
*/
|