| 
									
										
										
										
											2010-01-11 13:14:54 -08:00
										 |  |  |  | /* Copyright (c) 2009, 2010 Nicira Networks
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * 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 <config.h>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "ovsdb-data.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <assert.h>
 | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | #include <ctype.h>
 | 
					
						
							|  |  |  |  | #include <float.h>
 | 
					
						
							|  |  |  |  | #include <inttypes.h>
 | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | #include <limits.h>
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | #include "dynamic-string.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | #include "hash.h"
 | 
					
						
							|  |  |  |  | #include "ovsdb-error.h"
 | 
					
						
							|  |  |  |  | #include "json.h"
 | 
					
						
							|  |  |  |  | #include "shash.h"
 | 
					
						
							|  |  |  |  | #include "sort.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | #include "unicode.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct json * | 
					
						
							|  |  |  |  | wrap_json(const char *name, struct json *wrapped) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return json_array_create_2(json_string_create(name), wrapped); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_atom_init_default(union ovsdb_atom *atom, enum ovsdb_atomic_type type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         atom->integer = 0; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         atom->real = 0.0; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         atom->boolean = false; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         atom->string = xmemdup("", 1); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         uuid_zero(&atom->uuid); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-11 13:14:54 -08:00
										 |  |  |  | bool | 
					
						
							|  |  |  |  | ovsdb_atom_is_default(const union ovsdb_atom *atom, | 
					
						
							|  |  |  |  |                       enum ovsdb_atomic_type type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         return atom->integer == 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         return atom->real == 0.0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         return atom->boolean == false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         return atom->string[0] == '\0'; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         return uuid_is_zero(&atom->uuid); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_atom_clone(union ovsdb_atom *new, const union ovsdb_atom *old, | 
					
						
							|  |  |  |  |                  enum ovsdb_atomic_type type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         new->integer = old->integer; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         new->real = old->real; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         new->boolean = old->boolean; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         new->string = xstrdup(old->string); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         new->uuid = old->uuid; | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_atom_swap(union ovsdb_atom *a, union ovsdb_atom *b) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     union ovsdb_atom tmp = *a; | 
					
						
							|  |  |  |  |     *a = *b; | 
					
						
							|  |  |  |  |     *b = tmp; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | uint32_t | 
					
						
							|  |  |  |  | ovsdb_atom_hash(const union ovsdb_atom *atom, enum ovsdb_atomic_type type, | 
					
						
							|  |  |  |  |                 uint32_t basis) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         return hash_int(atom->integer, basis); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         return hash_double(atom->real, basis); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         return hash_boolean(atom->boolean, basis); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         return hash_string(atom->string, basis); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         return hash_int(uuid_hash(&atom->uuid), basis); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int | 
					
						
							|  |  |  |  | ovsdb_atom_compare_3way(const union ovsdb_atom *a, | 
					
						
							|  |  |  |  |                         const union ovsdb_atom *b, | 
					
						
							|  |  |  |  |                         enum ovsdb_atomic_type type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         return a->integer < b->integer ? -1 : a->integer > b->integer; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         return a->real < b->real ? -1 : a->real > b->real; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         return a->boolean - b->boolean; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         return strcmp(a->string, b->string); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         return uuid_compare_3way(&a->uuid, &b->uuid); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | unwrap_json(const struct json *json, const char *name, | 
					
						
							|  |  |  |  |             enum json_type value_type, const struct json **value) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (json->type != JSON_ARRAY | 
					
						
							|  |  |  |  |         || json->u.array.n != 2 | 
					
						
							|  |  |  |  |         || json->u.array.elems[0]->type != JSON_STRING | 
					
						
							|  |  |  |  |         || (name && strcmp(json->u.array.elems[0]->u.string, name)) | 
					
						
							|  |  |  |  |         || json->u.array.elems[1]->type != value_type) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         return ovsdb_syntax_error(json, NULL, "expected [\"%s\", <%s>]", name, | 
					
						
							|  |  |  |  |                                   json_type_to_string(value_type)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     *value = json->u.array.elems[1]; | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | parse_json_pair(const struct json *json, | 
					
						
							|  |  |  |  |                 const struct json **elem0, const struct json **elem1) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (json->type != JSON_ARRAY || json->u.array.n != 2) { | 
					
						
							|  |  |  |  |         return ovsdb_syntax_error(json, NULL, "expected 2-element array"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     *elem0 = json->u.array.elems[0]; | 
					
						
							|  |  |  |  |     *elem1 = json->u.array.elems[1]; | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  | static struct ovsdb_error * WARN_UNUSED_RESULT | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | ovsdb_atom_parse_uuid(struct uuid *uuid, const struct json *json, | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  |                       struct ovsdb_symbol_table *symtab) | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_error *error0; | 
					
						
							|  |  |  |  |     const struct json *value; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error0 = unwrap_json(json, "uuid", JSON_STRING, &value); | 
					
						
							|  |  |  |  |     if (!error0) { | 
					
						
							|  |  |  |  |         const char *uuid_string = json_string(value); | 
					
						
							|  |  |  |  |         if (!uuid_from_string(uuid, uuid_string)) { | 
					
						
							|  |  |  |  |             return ovsdb_syntax_error(json, NULL, "\"%s\" is not a valid UUID", | 
					
						
							|  |  |  |  |                                       uuid_string); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } else if (symtab) { | 
					
						
							|  |  |  |  |         struct ovsdb_error *error1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error1 = unwrap_json(json, "named-uuid", JSON_STRING, &value); | 
					
						
							|  |  |  |  |         if (!error1) { | 
					
						
							|  |  |  |  |             const char *name = json_string(value); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             ovsdb_error_destroy(error0); | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  |             *uuid = ovsdb_symbol_table_insert(symtab, name)->uuid; | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         ovsdb_error_destroy(error1); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return error0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | static struct ovsdb_error * WARN_UNUSED_RESULT | 
					
						
							|  |  |  |  | ovsdb_atom_from_json__(union ovsdb_atom *atom, enum ovsdb_atomic_type type, | 
					
						
							|  |  |  |  |                        const struct json *json, | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  |                        struct ovsdb_symbol_table *symtab) | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         if (json->type == JSON_INTEGER) { | 
					
						
							|  |  |  |  |             atom->integer = json->u.integer; | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         if (json->type == JSON_INTEGER) { | 
					
						
							|  |  |  |  |             atom->real = json->u.integer; | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } else if (json->type == JSON_REAL) { | 
					
						
							|  |  |  |  |             atom->real = json->u.real; | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         if (json->type == JSON_TRUE) { | 
					
						
							|  |  |  |  |             atom->boolean = true; | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } else if (json->type == JSON_FALSE) { | 
					
						
							|  |  |  |  |             atom->boolean = false; | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         if (json->type == JSON_STRING) { | 
					
						
							|  |  |  |  |             atom->string = xstrdup(json->u.string); | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         return ovsdb_atom_parse_uuid(&atom->uuid, json, symtab); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ovsdb_syntax_error(json, NULL, "expected %s", | 
					
						
							|  |  |  |  |                               ovsdb_atomic_type_to_string(type)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_atom_from_json(union ovsdb_atom *atom, | 
					
						
							|  |  |  |  |                      const struct ovsdb_base_type *base, | 
					
						
							|  |  |  |  |                      const struct json *json, | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  |                      struct ovsdb_symbol_table *symtab) | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error = ovsdb_atom_from_json__(atom, base->type, json, symtab); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error = ovsdb_atom_check_constraints(atom, base); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         ovsdb_atom_destroy(atom, base->type); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return error; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | struct json * | 
					
						
							|  |  |  |  | ovsdb_atom_to_json(const union ovsdb_atom *atom, enum ovsdb_atomic_type type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         return json_integer_create(atom->integer); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         return json_real_create(atom->real); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         return json_boolean_create(atom->boolean); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         return json_string_create(atom->string); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         return wrap_json("uuid", json_string_create_nocopy( | 
					
						
							|  |  |  |  |                              xasprintf(UUID_FMT, UUID_ARGS(&atom->uuid)))); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | static char * | 
					
						
							|  |  |  |  | ovsdb_atom_from_string__(union ovsdb_atom *atom, enum ovsdb_atomic_type type, | 
					
						
							|  |  |  |  |                          const char *s) | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: { | 
					
						
							|  |  |  |  |         long long int integer; | 
					
						
							|  |  |  |  |         if (!str_to_llong(s, 10, &integer)) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             return xasprintf("\"%s\" is not a valid integer", s); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         atom->integer = integer; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         if (!str_to_double(s, &atom->real)) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             return xasprintf("\"%s\" is not a valid real number", s); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-27 15:38:29 -08:00
										 |  |  |  |         /* Our JSON input routines map negative zero to zero, so do that here
 | 
					
						
							|  |  |  |  |          * too for consistency. */ | 
					
						
							|  |  |  |  |         if (atom->real == 0.0) { | 
					
						
							|  |  |  |  |             atom->real = 0.0; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         if (!strcmp(s, "true") || !strcmp(s, "yes") || !strcmp(s, "on") | 
					
						
							|  |  |  |  |             || !strcmp(s, "1")) { | 
					
						
							|  |  |  |  |             atom->boolean = true; | 
					
						
							|  |  |  |  |         } else if (!strcmp(s, "false") || !strcmp(s, "no") || !strcmp(s, "off") | 
					
						
							|  |  |  |  |                    || !strcmp(s, "0")) { | 
					
						
							|  |  |  |  |             atom->boolean = false; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             return xasprintf("\"%s\" is not a valid boolean " | 
					
						
							|  |  |  |  |                              "(use \"true\" or \"false\")", s); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         if (*s == '\0') { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             return xstrdup("An empty string is not valid as input; " | 
					
						
							|  |  |  |  |                            "use \"\" to represent the empty string"); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } else if (*s == '"') { | 
					
						
							|  |  |  |  |             size_t s_len = strlen(s); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (s_len < 2 || s[s_len - 1] != '"') { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |                 return xasprintf("%s: missing quote at end of " | 
					
						
							|  |  |  |  |                                  "quoted string", s); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |             } else if (!json_string_unescape(s + 1, s_len - 2, | 
					
						
							|  |  |  |  |                                              &atom->string)) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |                 char *error = xasprintf("%s: %s", s, atom->string); | 
					
						
							|  |  |  |  |                 free(atom->string); | 
					
						
							|  |  |  |  |                 return error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             atom->string = xstrdup(s); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         if (!uuid_from_string(&atom->uuid, s)) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             return xasprintf("\"%s\" is not a valid UUID", s); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | /* Initializes 'atom' to a value of type 'base' parsed from 's', which takes
 | 
					
						
							|  |  |  |  |  * one of the following forms: | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  *      - OVSDB_TYPE_INTEGER: A decimal integer optionally preceded by a sign. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  *      - OVSDB_TYPE_REAL: A floating-point number in the format accepted by | 
					
						
							|  |  |  |  |  *        strtod(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  *      - OVSDB_TYPE_BOOLEAN: "true", "yes", "on", "1" for true, or "false", | 
					
						
							|  |  |  |  |  *        "no", "off", or "0" for false. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  *      - OVSDB_TYPE_STRING: A JSON string if it begins with a quote, otherwise | 
					
						
							|  |  |  |  |  *        an arbitrary string. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  *      - OVSDB_TYPE_UUID: A UUID in RFC 4122 format. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns a null pointer if successful, otherwise an error message describing | 
					
						
							|  |  |  |  |  * the problem.  The caller is responsible for freeing the error. | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | char * | 
					
						
							|  |  |  |  | ovsdb_atom_from_string(union ovsdb_atom *atom, | 
					
						
							|  |  |  |  |                        const struct ovsdb_base_type *base, const char *s) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     char *msg; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     msg = ovsdb_atom_from_string__(atom, base->type, s); | 
					
						
							|  |  |  |  |     if (msg) { | 
					
						
							|  |  |  |  |         return msg; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error = ovsdb_atom_check_constraints(atom, base); | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         msg = ovsdb_error_to_string(error); | 
					
						
							|  |  |  |  |         ovsdb_error_destroy(error); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return msg; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | static bool | 
					
						
							|  |  |  |  | string_needs_quotes(const char *s) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const char *p = s; | 
					
						
							|  |  |  |  |     unsigned char c; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     c = *p++; | 
					
						
							|  |  |  |  |     if (!isalpha(c) && c != '_') { | 
					
						
							|  |  |  |  |         return true; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     while ((c = *p++) != '\0') { | 
					
						
							|  |  |  |  |         if (!isalpha(c) && c != '_' && c != '-' && c != '.') { | 
					
						
							|  |  |  |  |             return true; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!strcmp(s, "true") || !strcmp(s, "false")) { | 
					
						
							|  |  |  |  |         return true; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return false; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Appends 'atom' (which has the given 'type') to 'out', in a format acceptable
 | 
					
						
							|  |  |  |  |  * to ovsdb_atom_from_string().  */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_atom_to_string(const union ovsdb_atom *atom, enum ovsdb_atomic_type type, | 
					
						
							|  |  |  |  |                      struct ds *out) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         ds_put_format(out, "%"PRId64, atom->integer); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         ds_put_format(out, "%.*g", DBL_DIG, atom->real); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         ds_put_cstr(out, atom->boolean ? "true" : "false"); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         if (string_needs_quotes(atom->string)) { | 
					
						
							|  |  |  |  |             struct json json; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             json.type = JSON_STRING; | 
					
						
							|  |  |  |  |             json.u.string = atom->string; | 
					
						
							|  |  |  |  |             json_to_ds(&json, 0, out); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             ds_put_cstr(out, atom->string); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         ds_put_format(out, UUID_FMT, UUID_ARGS(&atom->uuid)); | 
					
						
							|  |  |  |  |         break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct ovsdb_error * | 
					
						
							|  |  |  |  | check_string_constraints(const char *s, | 
					
						
							|  |  |  |  |                          const struct ovsdb_string_constraints *c) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t n_chars; | 
					
						
							|  |  |  |  |     char *msg; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     msg = utf8_validate(s, &n_chars); | 
					
						
							|  |  |  |  |     if (msg) { | 
					
						
							|  |  |  |  |         struct ovsdb_error *error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                             "\"%s\" is not a valid UTF-8 string: %s", | 
					
						
							|  |  |  |  |                             s, msg); | 
					
						
							|  |  |  |  |         free(msg); | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (n_chars < c->minLen) { | 
					
						
							|  |  |  |  |         return ovsdb_error( | 
					
						
							|  |  |  |  |             "constraint violation", | 
					
						
							|  |  |  |  |             "\"%s\" length %zu is less than minimum allowed " | 
					
						
							|  |  |  |  |             "length %u", s, n_chars, c->minLen); | 
					
						
							|  |  |  |  |     } else if (n_chars > c->maxLen) { | 
					
						
							|  |  |  |  |         return ovsdb_error( | 
					
						
							|  |  |  |  |             "constraint violation", | 
					
						
							|  |  |  |  |             "\"%s\" length %zu is greater than maximum allowed " | 
					
						
							|  |  |  |  |             "length %u", s, n_chars, c->maxLen); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #if HAVE_PCRE
 | 
					
						
							|  |  |  |  |     if (c->re) { | 
					
						
							|  |  |  |  |         int retval; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         retval = pcre_exec(c->re, NULL, s, strlen(s), 0, | 
					
						
							|  |  |  |  |                            PCRE_ANCHORED | PCRE_NO_UTF8_CHECK, NULL, 0); | 
					
						
							|  |  |  |  |         if (retval == PCRE_ERROR_NOMATCH) { | 
					
						
							|  |  |  |  |             if (c->reComment) { | 
					
						
							|  |  |  |  |                 return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                    "\"%s\" is not a %s", s, c->reComment); | 
					
						
							|  |  |  |  |             } else { | 
					
						
							|  |  |  |  |                 return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                    "\"%s\" does not match regular expression " | 
					
						
							|  |  |  |  |                                    "/%s/", s, c->reMatch); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } else if (retval < 0) { | 
					
						
							|  |  |  |  |             /* PCRE doesn't have a function to translate an error code to a
 | 
					
						
							|  |  |  |  |              * description.  Bizarre.  See pcreapi(3) for error details. */ | 
					
						
							|  |  |  |  |             return ovsdb_error("internal error", "PCRE returned error %d", | 
					
						
							|  |  |  |  |                                retval); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | #endif  /* HAVE_PCRE */
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Checks whether 'atom' meets the constraints (if any) defined in 'base'.
 | 
					
						
							|  |  |  |  |  * (base->type must specify 'atom''s type.)  Returns a null pointer if the | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:41 -08:00
										 |  |  |  |  * constraints are met, otherwise an error that explains the violation. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Checking UUID constraints is deferred to transaction commit time, so this | 
					
						
							|  |  |  |  |  * function does nothing for UUID constraints. */ | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_atom_check_constraints(const union ovsdb_atom *atom, | 
					
						
							|  |  |  |  |                              const struct ovsdb_base_type *base) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (base->type) { | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_VOID: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_INTEGER: | 
					
						
							|  |  |  |  |         if (atom->integer >= base->u.integer.min | 
					
						
							|  |  |  |  |             && atom->integer <= base->u.integer.max) { | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } else if (base->u.integer.min != INT64_MIN) { | 
					
						
							|  |  |  |  |             if (base->u.integer.max != INT64_MAX) { | 
					
						
							|  |  |  |  |                 return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                    "%"PRId64" is not in the valid range " | 
					
						
							|  |  |  |  |                                    "%"PRId64" to %"PRId64" (inclusive)", | 
					
						
							|  |  |  |  |                                    atom->integer, | 
					
						
							|  |  |  |  |                                    base->u.integer.min, base->u.integer.max); | 
					
						
							|  |  |  |  |             } else { | 
					
						
							|  |  |  |  |                 return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                    "%"PRId64" is less than minimum allowed " | 
					
						
							|  |  |  |  |                                    "value %"PRId64, | 
					
						
							|  |  |  |  |                                    atom->integer, base->u.integer.min); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                "%"PRId64" is greater than maximum allowed " | 
					
						
							|  |  |  |  |                                "value %"PRId64, | 
					
						
							|  |  |  |  |                                atom->integer, base->u.integer.max); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_REAL: | 
					
						
							|  |  |  |  |         if (atom->real >= base->u.real.min && atom->real <= base->u.real.max) { | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         } else if (base->u.real.min != -DBL_MAX) { | 
					
						
							|  |  |  |  |             if (base->u.real.max != DBL_MAX) { | 
					
						
							|  |  |  |  |                 return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                    "%.*g is not in the valid range " | 
					
						
							|  |  |  |  |                                    "%.*g to %.*g (inclusive)", | 
					
						
							|  |  |  |  |                                    DBL_DIG, atom->real, | 
					
						
							|  |  |  |  |                                    DBL_DIG, base->u.real.min, | 
					
						
							|  |  |  |  |                                    DBL_DIG, base->u.real.max); | 
					
						
							|  |  |  |  |             } else { | 
					
						
							|  |  |  |  |                 return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                    "%.*g is less than minimum allowed " | 
					
						
							|  |  |  |  |                                    "value %.*g", | 
					
						
							|  |  |  |  |                                    DBL_DIG, atom->real, | 
					
						
							|  |  |  |  |                                    DBL_DIG, base->u.real.min); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             return ovsdb_error("constraint violation", | 
					
						
							|  |  |  |  |                                "%.*g is greater than maximum allowed " | 
					
						
							|  |  |  |  |                                "value %.*g", | 
					
						
							|  |  |  |  |                                DBL_DIG, atom->real, | 
					
						
							|  |  |  |  |                                DBL_DIG, base->u.real.max); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_BOOLEAN: | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_STRING: | 
					
						
							|  |  |  |  |         return check_string_constraints(atom->string, &base->u.string); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_TYPE_UUID: | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case OVSDB_N_TYPES: | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |         NOT_REACHED(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | static union ovsdb_atom * | 
					
						
							|  |  |  |  | alloc_default_atoms(enum ovsdb_atomic_type type, size_t n) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (type != OVSDB_TYPE_VOID && n) { | 
					
						
							|  |  |  |  |         union ovsdb_atom *atoms; | 
					
						
							|  |  |  |  |         unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         atoms = xmalloc(n * sizeof *atoms); | 
					
						
							|  |  |  |  |         for (i = 0; i < n; i++) { | 
					
						
							|  |  |  |  |             ovsdb_atom_init_default(&atoms[i], type); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         return atoms; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         /* Avoid wasting memory in the n == 0 case, because xmalloc(0) is
 | 
					
						
							|  |  |  |  |          * treated as xmalloc(1). */ | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_init_empty(struct ovsdb_datum *datum) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     datum->n = 0; | 
					
						
							|  |  |  |  |     datum->keys = NULL; | 
					
						
							|  |  |  |  |     datum->values = NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_init_default(struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                          const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     datum->n = type->n_min; | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     datum->keys = alloc_default_atoms(type->key.type, datum->n); | 
					
						
							|  |  |  |  |     datum->values = alloc_default_atoms(type->value.type, datum->n); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-11 13:14:54 -08:00
										 |  |  |  | bool | 
					
						
							|  |  |  |  | ovsdb_datum_is_default(const struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                        const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (datum->n != type->n_min) { | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     for (i = 0; i < datum->n; i++) { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         if (!ovsdb_atom_is_default(&datum->keys[i], type->key.type)) { | 
					
						
							| 
									
										
										
										
											2010-01-11 13:14:54 -08:00
										 |  |  |  |             return false; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         if (type->value.type != OVSDB_TYPE_VOID | 
					
						
							|  |  |  |  |             && !ovsdb_atom_is_default(&datum->values[i], type->value.type)) { | 
					
						
							| 
									
										
										
										
											2010-01-11 13:14:54 -08:00
										 |  |  |  |             return false; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | static union ovsdb_atom * | 
					
						
							|  |  |  |  | clone_atoms(const union ovsdb_atom *old, enum ovsdb_atomic_type type, size_t n) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (type != OVSDB_TYPE_VOID && n) { | 
					
						
							|  |  |  |  |         union ovsdb_atom *new; | 
					
						
							|  |  |  |  |         unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         new = xmalloc(n * sizeof *new); | 
					
						
							|  |  |  |  |         for (i = 0; i < n; i++) { | 
					
						
							|  |  |  |  |             ovsdb_atom_clone(&new[i], &old[i], type); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         return new; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         /* Avoid wasting memory in the n == 0 case, because xmalloc(0) is
 | 
					
						
							|  |  |  |  |          * treated as xmalloc(1). */ | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_clone(struct ovsdb_datum *new, const struct ovsdb_datum *old, | 
					
						
							|  |  |  |  |                   const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int n = old->n; | 
					
						
							|  |  |  |  |     new->n = n; | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     new->keys = clone_atoms(old->keys, type->key.type, n); | 
					
						
							|  |  |  |  |     new->values = clone_atoms(old->values, type->value.type, n); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | free_data(enum ovsdb_atomic_type type, | 
					
						
							|  |  |  |  |           union ovsdb_atom *atoms, size_t n_atoms) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (ovsdb_atom_needs_destruction(type)) { | 
					
						
							|  |  |  |  |         unsigned int i; | 
					
						
							|  |  |  |  |         for (i = 0; i < n_atoms; i++) { | 
					
						
							|  |  |  |  |             ovsdb_atom_destroy(&atoms[i], type); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     free(atoms); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_destroy(struct ovsdb_datum *datum, const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     free_data(type->key.type, datum->keys, datum->n); | 
					
						
							|  |  |  |  |     free_data(type->value.type, datum->values, datum->n); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_swap(struct ovsdb_datum *a, struct ovsdb_datum *b) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_datum tmp = *a; | 
					
						
							|  |  |  |  |     *a = *b; | 
					
						
							|  |  |  |  |     *b = tmp; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct ovsdb_datum_sort_cbdata { | 
					
						
							|  |  |  |  |     const struct ovsdb_type *type; | 
					
						
							|  |  |  |  |     struct ovsdb_datum *datum; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | ovsdb_datum_sort_compare_cb(size_t a, size_t b, void *cbdata_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_datum_sort_cbdata *cbdata = cbdata_; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ovsdb_atom_compare_3way(&cbdata->datum->keys[a], | 
					
						
							|  |  |  |  |                                    &cbdata->datum->keys[b], | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                                    cbdata->type->key.type); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | ovsdb_datum_sort_swap_cb(size_t a, size_t b, void *cbdata_) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_datum_sort_cbdata *cbdata = cbdata_; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovsdb_atom_swap(&cbdata->datum->keys[a], &cbdata->datum->keys[b]); | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     if (cbdata->type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |         ovsdb_atom_swap(&cbdata->datum->values[a], &cbdata->datum->values[b]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | struct ovsdb_error * | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | ovsdb_datum_sort(struct ovsdb_datum *datum, const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (datum->n < 2) { | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         struct ovsdb_datum_sort_cbdata cbdata; | 
					
						
							|  |  |  |  |         size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         cbdata.type = type; | 
					
						
							|  |  |  |  |         cbdata.datum = datum; | 
					
						
							|  |  |  |  |         sort(datum->n, ovsdb_datum_sort_compare_cb, ovsdb_datum_sort_swap_cb, | 
					
						
							|  |  |  |  |              &cbdata); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         for (i = 0; i < datum->n - 1; i++) { | 
					
						
							|  |  |  |  |             if (ovsdb_atom_equals(&datum->keys[i], &datum->keys[i + 1], | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                                   type->key.type)) { | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |                 if (ovsdb_type_is_map(type)) { | 
					
						
							|  |  |  |  |                     return ovsdb_error(NULL, "map contains duplicate key"); | 
					
						
							|  |  |  |  |                 } else { | 
					
						
							|  |  |  |  |                     return ovsdb_error(NULL, "set contains duplicate"); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | /* Checks that each of the atoms in 'datum' conforms to the constraints
 | 
					
						
							|  |  |  |  |  * specified by its 'type'.  Returns an error if a constraint is violated, | 
					
						
							|  |  |  |  |  * otherwise a null pointer. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This function is not commonly useful because the most ordinary way to obtain | 
					
						
							|  |  |  |  |  * a datum is ultimately via ovsdb_atom_from_string() or | 
					
						
							|  |  |  |  |  * ovsdb_atom_from_json(), which check constraints themselves. */ | 
					
						
							|  |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_datum_check_constraints(const struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                               const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < datum->n; i++) { | 
					
						
							|  |  |  |  |         error = ovsdb_atom_check_constraints(&datum->keys[i], &type->key); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             return error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							|  |  |  |  |         for (i = 0; i < datum->n; i++) { | 
					
						
							|  |  |  |  |             error = ovsdb_atom_check_constraints(&datum->values[i], | 
					
						
							|  |  |  |  |                                                  &type->value); | 
					
						
							|  |  |  |  |             if (error) { | 
					
						
							|  |  |  |  |                 return error; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | struct ovsdb_error * | 
					
						
							|  |  |  |  | ovsdb_datum_from_json(struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                       const struct ovsdb_type *type, | 
					
						
							|  |  |  |  |                       const struct json *json, | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  |                       struct ovsdb_symbol_table *symtab) | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_error *error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (ovsdb_type_is_scalar(type)) { | 
					
						
							|  |  |  |  |         datum->n = 1; | 
					
						
							|  |  |  |  |         datum->keys = xmalloc(sizeof *datum->keys); | 
					
						
							|  |  |  |  |         datum->values = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         error = ovsdb_atom_from_json(&datum->keys[0], &type->key, | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |                                      json, symtab); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             free(datum->keys); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         bool is_map = ovsdb_type_is_map(type); | 
					
						
							|  |  |  |  |         const char *class = is_map ? "map" : "set"; | 
					
						
							|  |  |  |  |         const struct json *inner; | 
					
						
							|  |  |  |  |         unsigned int i; | 
					
						
							|  |  |  |  |         size_t n; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         assert(is_map || ovsdb_type_is_set(type)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = unwrap_json(json, class, JSON_ARRAY, &inner); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             return error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         n = inner->u.array.n; | 
					
						
							|  |  |  |  |         if (n < type->n_min || n > type->n_max) { | 
					
						
							|  |  |  |  |             return ovsdb_syntax_error(json, NULL, "%s must have %u to " | 
					
						
							|  |  |  |  |                                       "%u members but %zu are present", | 
					
						
							|  |  |  |  |                                       class, type->n_min, type->n_max, n); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         datum->n = 0; | 
					
						
							|  |  |  |  |         datum->keys = xmalloc(n * sizeof *datum->keys); | 
					
						
							|  |  |  |  |         datum->values = is_map ? xmalloc(n * sizeof *datum->values) : NULL; | 
					
						
							|  |  |  |  |         for (i = 0; i < n; i++) { | 
					
						
							|  |  |  |  |             const struct json *element = inner->u.array.elems[i]; | 
					
						
							|  |  |  |  |             const struct json *key = NULL; | 
					
						
							|  |  |  |  |             const struct json *value = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (!is_map) { | 
					
						
							|  |  |  |  |                 key = element; | 
					
						
							|  |  |  |  |             } else { | 
					
						
							|  |  |  |  |                 error = parse_json_pair(element, &key, &value); | 
					
						
							|  |  |  |  |                 if (error) { | 
					
						
							|  |  |  |  |                     goto error; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |             error = ovsdb_atom_from_json(&datum->keys[i], &type->key, | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |                                          key, symtab); | 
					
						
							|  |  |  |  |             if (error) { | 
					
						
							|  |  |  |  |                 goto error; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (is_map) { | 
					
						
							|  |  |  |  |                 error = ovsdb_atom_from_json(&datum->values[i], | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                                              &type->value, value, symtab); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |                 if (error) { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                     ovsdb_atom_destroy(&datum->keys[i], type->key.type); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |                     goto error; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             datum->n++; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         error = ovsdb_datum_sort(datum, type); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     error: | 
					
						
							|  |  |  |  |         ovsdb_datum_destroy(datum, type); | 
					
						
							|  |  |  |  |         return error; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct json * | 
					
						
							|  |  |  |  | ovsdb_datum_to_json(const struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                     const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     /* These tests somewhat tolerate a 'datum' that does not exactly match
 | 
					
						
							|  |  |  |  |      * 'type', in particular a datum with 'n' not in the allowed range. */ | 
					
						
							|  |  |  |  |     if (datum->n == 1 && ovsdb_type_is_scalar(type)) { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         return ovsdb_atom_to_json(&datum->keys[0], type->key.type); | 
					
						
							|  |  |  |  |     } else if (type->value.type == OVSDB_TYPE_VOID) { | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |         struct json **elems; | 
					
						
							|  |  |  |  |         size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         elems = xmalloc(datum->n * sizeof *elems); | 
					
						
							|  |  |  |  |         for (i = 0; i < datum->n; i++) { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |             elems[i] = ovsdb_atom_to_json(&datum->keys[i], type->key.type); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return wrap_json("set", json_array_create(elems, datum->n)); | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         struct json **elems; | 
					
						
							|  |  |  |  |         size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         elems = xmalloc(datum->n * sizeof *elems); | 
					
						
							|  |  |  |  |         for (i = 0; i < datum->n; i++) { | 
					
						
							|  |  |  |  |             elems[i] = json_array_create_2( | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                 ovsdb_atom_to_json(&datum->keys[i], type->key.type), | 
					
						
							|  |  |  |  |                 ovsdb_atom_to_json(&datum->values[i], type->value.type)); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return wrap_json("map", json_array_create(elems, datum->n)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | static const char * | 
					
						
							|  |  |  |  | skip_spaces(const char *p) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-27 15:58:32 -08:00
										 |  |  |  |     while (isspace((unsigned char) *p)) { | 
					
						
							|  |  |  |  |         p++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return p; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  | parse_atom_token(const char **s, const struct ovsdb_base_type *base, | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |                  union ovsdb_atom *atom) | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     char *token, *error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     error = ovsdb_token_parse(s, &token); | 
					
						
							|  |  |  |  |     if (!error) { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         error = ovsdb_atom_from_string(atom, base, token); | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         free(token); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return error; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  | static char * | 
					
						
							|  |  |  |  | parse_key_value(const char **s, const struct ovsdb_type *type, | 
					
						
							|  |  |  |  |                 union ovsdb_atom *key, union ovsdb_atom *value) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const char *start = *s; | 
					
						
							|  |  |  |  |     char *error; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     error = parse_atom_token(s, &type->key, key); | 
					
						
							|  |  |  |  |     if (!error && type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							| 
									
										
										
										
											2010-01-28 10:09:15 -08:00
										 |  |  |  |         *s = skip_spaces(*s); | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         if (**s == '=') { | 
					
						
							|  |  |  |  |             (*s)++; | 
					
						
							| 
									
										
										
										
											2010-01-28 10:09:15 -08:00
										 |  |  |  |             *s = skip_spaces(*s); | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |             error = parse_atom_token(s, &type->value, value); | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |             error = xasprintf("%s: syntax error at \"%c\" expecting \"=\"", | 
					
						
							|  |  |  |  |                               start, **s); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |             ovsdb_atom_destroy(key, type->key.type); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | free_key_value(const struct ovsdb_type *type, | 
					
						
							|  |  |  |  |                union ovsdb_atom *key, union ovsdb_atom *value) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     ovsdb_atom_destroy(key, type->key.type); | 
					
						
							|  |  |  |  |     if (type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							|  |  |  |  |         ovsdb_atom_destroy(value, type->value.type); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Initializes 'datum' as a datum of the given 'type', parsing its contents
 | 
					
						
							|  |  |  |  |  * from 's'.  The format of 's' is a series of space or comma separated atoms | 
					
						
							|  |  |  |  |  * or, for a map, '='-delimited pairs of atoms.  Each atom must in a format | 
					
						
							|  |  |  |  |  * acceptable to ovsdb_atom_from_string().  Optionally, a set may be enclosed | 
					
						
							|  |  |  |  |  * in "[]" or a map in "{}"; for an empty set or map these punctuators are | 
					
						
							|  |  |  |  |  * required. */ | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  | char * | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | ovsdb_datum_from_string(struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                         const struct ovsdb_type *type, const char *s) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     bool is_map = ovsdb_type_is_map(type); | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     struct ovsdb_error *dberror; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |     const char *p; | 
					
						
							|  |  |  |  |     int end_delim; | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     char *error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ovsdb_datum_init_empty(datum); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Swallow a leading delimiter if there is one. */ | 
					
						
							|  |  |  |  |     p = skip_spaces(s); | 
					
						
							|  |  |  |  |     if (*p == (is_map ? '{' : '[')) { | 
					
						
							|  |  |  |  |         end_delim = is_map ? '}' : ']'; | 
					
						
							|  |  |  |  |         p = skip_spaces(p + 1); | 
					
						
							|  |  |  |  |     } else if (!*p) { | 
					
						
							|  |  |  |  |         if (is_map) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             return xstrdup("use \"{}\" to specify the empty map"); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             return xstrdup("use \"[]\" to specify the empty set"); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         end_delim = 0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     while (*p && *p != end_delim) { | 
					
						
							|  |  |  |  |         union ovsdb_atom key, value; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (ovsdb_token_is_delim(*p)) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             error = xasprintf("%s: unexpected \"%c\" parsing %s", | 
					
						
							|  |  |  |  |                               s, *p, ovsdb_type_to_english(type)); | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* Add to datum. */ | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         error = parse_key_value(&p, type, &key, &value); | 
					
						
							|  |  |  |  |         if (error) { | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         ovsdb_datum_add_unsafe(datum, &key, &value, type); | 
					
						
							|  |  |  |  |         free_key_value(type, &key, &value); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* Skip optional white space and comma. */ | 
					
						
							|  |  |  |  |         p = skip_spaces(p); | 
					
						
							|  |  |  |  |         if (*p == ',') { | 
					
						
							|  |  |  |  |             p = skip_spaces(p + 1); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (*p != end_delim) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         error = xasprintf("%s: missing \"%c\" at end of data", s, end_delim); | 
					
						
							|  |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     if (end_delim) { | 
					
						
							|  |  |  |  |         p = skip_spaces(p + 1); | 
					
						
							|  |  |  |  |         if (*p) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             error = xasprintf("%s: trailing garbage after \"%c\"", | 
					
						
							|  |  |  |  |                               s, end_delim); | 
					
						
							|  |  |  |  |             goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (datum->n < type->n_min) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         error = xasprintf("%s: %u %s specified but the minimum number is %u", | 
					
						
							|  |  |  |  |                           s, datum->n, is_map ? "pair(s)" : "value(s)", | 
					
						
							|  |  |  |  |                           type->n_min); | 
					
						
							|  |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |     } else if (datum->n > type->n_max) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         error = xasprintf("%s: %u %s specified but the maximum number is %u", | 
					
						
							|  |  |  |  |                           s, datum->n, is_map ? "pair(s)" : "value(s)", | 
					
						
							|  |  |  |  |             type->n_max); | 
					
						
							|  |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     dberror = ovsdb_datum_sort(datum, type); | 
					
						
							|  |  |  |  |     if (dberror) { | 
					
						
							|  |  |  |  |         ovsdb_error_destroy(dberror); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         if (ovsdb_type_is_map(type)) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             error = xasprintf("%s: map contains duplicate key", s); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |             error = xasprintf("%s: set contains duplicate value", s); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | error: | 
					
						
							|  |  |  |  |     ovsdb_datum_destroy(datum, type); | 
					
						
							|  |  |  |  |     ovsdb_datum_init_empty(datum); | 
					
						
							|  |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Appends to 'out' the 'datum' (with the given 'type') in a format acceptable
 | 
					
						
							|  |  |  |  |  * to ovsdb_datum_from_string(). */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_to_string(const struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                       const struct ovsdb_type *type, struct ds *out) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     bool is_map = ovsdb_type_is_map(type); | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (type->n_max > 1 || !datum->n) { | 
					
						
							|  |  |  |  |         ds_put_char(out, is_map ? '{' : '['); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     for (i = 0; i < datum->n; i++) { | 
					
						
							|  |  |  |  |         if (i > 0) { | 
					
						
							|  |  |  |  |             ds_put_cstr(out, ", "); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         ovsdb_atom_to_string(&datum->keys[i], type->key.type, out); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         if (is_map) { | 
					
						
							|  |  |  |  |             ds_put_char(out, '='); | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |             ovsdb_atom_to_string(&datum->values[i], type->value.type, out); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (type->n_max > 1 || !datum->n) { | 
					
						
							|  |  |  |  |         ds_put_char(out, is_map ? '}' : ']'); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | static uint32_t | 
					
						
							|  |  |  |  | hash_atoms(enum ovsdb_atomic_type type, const union ovsdb_atom *atoms, | 
					
						
							|  |  |  |  |            unsigned int n, uint32_t basis) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (type != OVSDB_TYPE_VOID) { | 
					
						
							|  |  |  |  |         unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         for (i = 0; i < n; i++) { | 
					
						
							|  |  |  |  |             basis = ovsdb_atom_hash(&atoms[i], type, basis); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return basis; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | uint32_t | 
					
						
							|  |  |  |  | ovsdb_datum_hash(const struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                  const struct ovsdb_type *type, uint32_t basis) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     basis = hash_atoms(type->key.type, datum->keys, datum->n, basis); | 
					
						
							|  |  |  |  |     basis ^= (type->key.type << 24) | (type->value.type << 16) | datum->n; | 
					
						
							|  |  |  |  |     basis = hash_atoms(type->value.type, datum->values, datum->n, basis); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |     return basis; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static int | 
					
						
							|  |  |  |  | atom_arrays_compare_3way(const union ovsdb_atom *a, | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |                          const union ovsdb_atom *b, | 
					
						
							|  |  |  |  |                          enum ovsdb_atomic_type type, | 
					
						
							|  |  |  |  |                          size_t n) | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < n; i++) { | 
					
						
							|  |  |  |  |         int cmp = ovsdb_atom_compare_3way(&a[i], &b[i], type); | 
					
						
							|  |  |  |  |         if (cmp) { | 
					
						
							|  |  |  |  |             return cmp; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | bool | 
					
						
							|  |  |  |  | ovsdb_datum_equals(const struct ovsdb_datum *a, | 
					
						
							|  |  |  |  |                    const struct ovsdb_datum *b, | 
					
						
							|  |  |  |  |                    const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return !ovsdb_datum_compare_3way(a, b, type); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int | 
					
						
							|  |  |  |  | ovsdb_datum_compare_3way(const struct ovsdb_datum *a, | 
					
						
							|  |  |  |  |                          const struct ovsdb_datum *b, | 
					
						
							|  |  |  |  |                          const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int cmp; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (a->n != b->n) { | 
					
						
							|  |  |  |  |         return a->n < b->n ? -1 : 1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     cmp = atom_arrays_compare_3way(a->keys, b->keys, type->key.type, a->n); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |     if (cmp) { | 
					
						
							|  |  |  |  |         return cmp; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     return (type->value.type == OVSDB_TYPE_VOID ? 0 | 
					
						
							|  |  |  |  |             : atom_arrays_compare_3way(a->values, b->values, type->value.type, | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |                                        a->n)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  | /* If 'key' is one of the keys in 'datum', returns its index within 'datum',
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |  * otherwise UINT_MAX.  'key.type' must be the type of the atoms stored in the | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |  * 'keys' array in 'datum'. | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | unsigned int | 
					
						
							|  |  |  |  | ovsdb_datum_find_key(const struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                      const union ovsdb_atom *key, | 
					
						
							|  |  |  |  |                      enum ovsdb_atomic_type key_type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int low = 0; | 
					
						
							|  |  |  |  |     unsigned int high = datum->n; | 
					
						
							|  |  |  |  |     while (low < high) { | 
					
						
							|  |  |  |  |         unsigned int idx = (low + high) / 2; | 
					
						
							|  |  |  |  |         int cmp = ovsdb_atom_compare_3way(key, &datum->keys[idx], key_type); | 
					
						
							|  |  |  |  |         if (cmp < 0) { | 
					
						
							|  |  |  |  |             high = idx; | 
					
						
							|  |  |  |  |         } else if (cmp > 0) { | 
					
						
							|  |  |  |  |             low = idx + 1; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             return idx; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return UINT_MAX; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* If 'key' and 'value' is one of the key-value pairs in 'datum', returns its
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |  * index within 'datum', otherwise UINT_MAX.  'key.type' must be the type of | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |  * the atoms stored in the 'keys' array in 'datum'.  'value_type' may be the | 
					
						
							|  |  |  |  |  * type of the 'values' atoms or OVSDB_TYPE_VOID to compare only keys. | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | unsigned int | 
					
						
							|  |  |  |  | ovsdb_datum_find_key_value(const struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                            const union ovsdb_atom *key, | 
					
						
							|  |  |  |  |                            enum ovsdb_atomic_type key_type, | 
					
						
							|  |  |  |  |                            const union ovsdb_atom *value, | 
					
						
							|  |  |  |  |                            enum ovsdb_atomic_type value_type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int idx = ovsdb_datum_find_key(datum, key, key_type); | 
					
						
							|  |  |  |  |     if (idx != UINT_MAX | 
					
						
							|  |  |  |  |         && value_type != OVSDB_TYPE_VOID | 
					
						
							|  |  |  |  |         && !ovsdb_atom_equals(&datum->values[idx], value, value_type)) { | 
					
						
							|  |  |  |  |         idx = UINT_MAX; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return idx; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | /* If atom 'i' in 'a' is also in 'b', returns its index in 'b', otherwise
 | 
					
						
							|  |  |  |  |  * UINT_MAX.  'type' must be the type of 'a' and 'b', except that | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |  * type->value.type may be set to OVSDB_TYPE_VOID to compare keys but not | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |  * values. */ | 
					
						
							|  |  |  |  | static unsigned int | 
					
						
							|  |  |  |  | ovsdb_datum_find(const struct ovsdb_datum *a, int i, | 
					
						
							|  |  |  |  |                  const struct ovsdb_datum *b, | 
					
						
							|  |  |  |  |                  const struct ovsdb_type *type) | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |     return ovsdb_datum_find_key_value(b, | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                                       &a->keys[i], type->key.type, | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |                                       a->values ? &a->values[i] : NULL, | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                                       type->value.type); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Returns true if every element in 'a' is also in 'b', false otherwise. */ | 
					
						
							|  |  |  |  | bool | 
					
						
							|  |  |  |  | ovsdb_datum_includes_all(const struct ovsdb_datum *a, | 
					
						
							|  |  |  |  |                          const struct ovsdb_datum *b, | 
					
						
							|  |  |  |  |                          const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < a->n; i++) { | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |         if (ovsdb_datum_find(a, i, b, type) == UINT_MAX) { | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |             return false; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Returns true if no element in 'a' is also in 'b', false otherwise. */ | 
					
						
							|  |  |  |  | bool | 
					
						
							|  |  |  |  | ovsdb_datum_excludes_all(const struct ovsdb_datum *a, | 
					
						
							|  |  |  |  |                          const struct ovsdb_datum *b, | 
					
						
							|  |  |  |  |                          const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < a->n; i++) { | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |         if (ovsdb_datum_find(a, i, b, type) != UINT_MAX) { | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |             return false; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | ovsdb_datum_reallocate(struct ovsdb_datum *a, const struct ovsdb_type *type, | 
					
						
							|  |  |  |  |                        unsigned int capacity) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     a->keys = xrealloc(a->keys, capacity * sizeof *a->keys); | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     if (type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |         a->values = xrealloc(a->values, capacity * sizeof *a->values); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  | /* Removes the element with index 'idx' from 'datum', which has type 'type'.
 | 
					
						
							|  |  |  |  |  * If 'idx' is not the last element in 'datum', then the removed element is | 
					
						
							|  |  |  |  |  * replaced by the (former) last element. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This function does not maintain ovsdb_datum invariants.  Use | 
					
						
							|  |  |  |  |  * ovsdb_datum_sort() to check and restore these invariants. */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_remove_unsafe(struct ovsdb_datum *datum, size_t idx, | 
					
						
							|  |  |  |  |                           const struct ovsdb_type *type) | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     ovsdb_atom_destroy(&datum->keys[idx], type->key.type); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |     datum->keys[idx] = datum->keys[datum->n - 1]; | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     if (type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							|  |  |  |  |         ovsdb_atom_destroy(&datum->values[idx], type->value.type); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |         datum->values[idx] = datum->values[datum->n - 1]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     datum->n--; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Adds the element with the given 'key' and 'value' to 'datum', which must
 | 
					
						
							|  |  |  |  |  * have the specified 'type'. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This function always allocates memory, so it is not an efficient way to add | 
					
						
							|  |  |  |  |  * a number of elements to a datum. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This function does not maintain ovsdb_datum invariants.  Use | 
					
						
							|  |  |  |  |  * ovsdb_datum_sort() to check and restore these invariants.  (But a datum with | 
					
						
							|  |  |  |  |  * 0 or 1 elements cannot violate the invariants anyhow.) */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_add_unsafe(struct ovsdb_datum *datum, | 
					
						
							|  |  |  |  |                        const union ovsdb_atom *key, | 
					
						
							|  |  |  |  |                        const union ovsdb_atom *value, | 
					
						
							|  |  |  |  |                        const struct ovsdb_type *type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t idx = datum->n++; | 
					
						
							|  |  |  |  |     datum->keys = xrealloc(datum->keys, datum->n * sizeof *datum->keys); | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     ovsdb_atom_clone(&datum->keys[idx], key, type->key.type); | 
					
						
							|  |  |  |  |     if (type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |         datum->values = xrealloc(datum->values, | 
					
						
							|  |  |  |  |                                  datum->n * sizeof *datum->values); | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         ovsdb_atom_clone(&datum->values[idx], value, type->value.type); | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  | ovsdb_datum_union(struct ovsdb_datum *a, const struct ovsdb_datum *b, | 
					
						
							|  |  |  |  |                   const struct ovsdb_type *type, bool replace) | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int n; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |     size_t bi; | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     n = a->n; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |     for (bi = 0; bi < b->n; bi++) { | 
					
						
							|  |  |  |  |         unsigned int ai; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         ai = ovsdb_datum_find_key(a, &b->keys[bi], type->key.type); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |         if (ai == UINT_MAX) { | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |             if (n == a->n) { | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |                 ovsdb_datum_reallocate(a, type, a->n + (b->n - bi)); | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |             ovsdb_atom_clone(&a->keys[n], &b->keys[bi], type->key.type); | 
					
						
							|  |  |  |  |             if (type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |                 ovsdb_atom_clone(&a->values[n], &b->values[bi], | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                                  type->value.type); | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |             n++; | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |         } else if (replace && type->value.type != OVSDB_TYPE_VOID) { | 
					
						
							|  |  |  |  |             ovsdb_atom_destroy(&a->values[ai], type->value.type); | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |             ovsdb_atom_clone(&a->values[ai], &b->values[bi], | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |                              type->value.type); | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (n != a->n) { | 
					
						
							|  |  |  |  |         struct ovsdb_error *error; | 
					
						
							|  |  |  |  |         a->n = n; | 
					
						
							|  |  |  |  |         error = ovsdb_datum_sort(a, type); | 
					
						
							|  |  |  |  |         assert(!error); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_datum_subtract(struct ovsdb_datum *a, const struct ovsdb_type *a_type, | 
					
						
							|  |  |  |  |                      const struct ovsdb_datum *b, | 
					
						
							|  |  |  |  |                      const struct ovsdb_type *b_type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     bool changed = false; | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 14:09:36 -08:00
										 |  |  |  |     assert(a_type->key.type == b_type->key.type); | 
					
						
							|  |  |  |  |     assert(a_type->value.type == b_type->value.type | 
					
						
							|  |  |  |  |            || b_type->value.type == OVSDB_TYPE_VOID); | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* XXX The big-O of this could easily be improved. */ | 
					
						
							|  |  |  |  |     for (i = 0; i < a->n; ) { | 
					
						
							|  |  |  |  |         unsigned int idx = ovsdb_datum_find(a, i, b, b_type); | 
					
						
							|  |  |  |  |         if (idx != UINT_MAX) { | 
					
						
							|  |  |  |  |             changed = true; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:08:57 -08:00
										 |  |  |  |             ovsdb_datum_remove_unsafe(a, i, a_type); | 
					
						
							| 
									
										
										
										
											2009-12-16 10:49:31 -08:00
										 |  |  |  |         } else { | 
					
						
							|  |  |  |  |             i++; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (changed) { | 
					
						
							|  |  |  |  |         struct ovsdb_error *error = ovsdb_datum_sort(a, a_type); | 
					
						
							|  |  |  |  |         assert(!error); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | struct ovsdb_symbol_table { | 
					
						
							|  |  |  |  |     struct shash sh; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct ovsdb_symbol_table * | 
					
						
							|  |  |  |  | ovsdb_symbol_table_create(void) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_symbol_table *symtab = xmalloc(sizeof *symtab); | 
					
						
							|  |  |  |  |     shash_init(&symtab->sh); | 
					
						
							|  |  |  |  |     return symtab; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | ovsdb_symbol_table_destroy(struct ovsdb_symbol_table *symtab) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (symtab) { | 
					
						
							|  |  |  |  |         struct shash_node *node, *next; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         SHASH_FOR_EACH_SAFE (node, next, &symtab->sh) { | 
					
						
							| 
									
										
										
										
											2009-12-07 11:47:48 -08:00
										 |  |  |  |             struct ovsdb_symbol *symbol = node->data; | 
					
						
							|  |  |  |  |             free(symbol); | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  |             shash_delete(&symtab->sh, node); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         shash_destroy(&symtab->sh); | 
					
						
							|  |  |  |  |         free(symtab); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-07 11:47:48 -08:00
										 |  |  |  | struct ovsdb_symbol * | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | ovsdb_symbol_table_get(const struct ovsdb_symbol_table *symtab, | 
					
						
							|  |  |  |  |                        const char *name) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return shash_find_data(&symtab->sh, name); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  | struct ovsdb_symbol * | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | ovsdb_symbol_table_put(struct ovsdb_symbol_table *symtab, const char *name, | 
					
						
							| 
									
										
										
										
											2009-12-07 11:47:48 -08:00
										 |  |  |  |                        const struct uuid *uuid, bool used) | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-12-07 11:47:48 -08:00
										 |  |  |  |     struct ovsdb_symbol *symbol; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     assert(!ovsdb_symbol_table_get(symtab, name)); | 
					
						
							|  |  |  |  |     symbol = xmalloc(sizeof *symbol); | 
					
						
							|  |  |  |  |     symbol->uuid = *uuid; | 
					
						
							|  |  |  |  |     symbol->used = used; | 
					
						
							|  |  |  |  |     shash_add(&symtab->sh, name, symbol); | 
					
						
							| 
									
										
										
										
											2010-02-08 16:03:21 -08:00
										 |  |  |  |     return symbol; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct ovsdb_symbol * | 
					
						
							|  |  |  |  | ovsdb_symbol_table_insert(struct ovsdb_symbol_table *symtab, | 
					
						
							|  |  |  |  |                           const char *name) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct ovsdb_symbol *symbol; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     symbol = ovsdb_symbol_table_get(symtab, name); | 
					
						
							|  |  |  |  |     if (!symbol) { | 
					
						
							|  |  |  |  |         struct uuid uuid; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         uuid_generate(&uuid); | 
					
						
							|  |  |  |  |         symbol = ovsdb_symbol_table_put(symtab, name, &uuid, false); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return symbol; | 
					
						
							| 
									
										
										
										
											2009-11-04 15:11:44 -08:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |  | 
					
						
							|  |  |  |  | /* Extracts a token from the beginning of 's' and returns a pointer just after
 | 
					
						
							|  |  |  |  |  * the token.  Stores the token itself into '*outp', which the caller is | 
					
						
							|  |  |  |  |  * responsible for freeing (with free()). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If 's[0]' is a delimiter, the returned token is the empty string. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * A token extends from 's' to the first delimiter, as defined by | 
					
						
							|  |  |  |  |  * ovsdb_token_is_delim(), or until the end of the string.  A delimiter can be | 
					
						
							|  |  |  |  |  * escaped with a backslash, in which case the backslash does not appear in the | 
					
						
							|  |  |  |  |  * output.  Double quotes also cause delimiters to be ignored, but the double | 
					
						
							|  |  |  |  |  * quotes are retained in the output.  (Backslashes inside double quotes are | 
					
						
							|  |  |  |  |  * not removed, either.) | 
					
						
							|  |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  | char * | 
					
						
							|  |  |  |  | ovsdb_token_parse(const char **s, char **outp) | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     const char *p; | 
					
						
							|  |  |  |  |     struct ds out; | 
					
						
							|  |  |  |  |     bool in_quotes; | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     char *error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     ds_init(&out); | 
					
						
							|  |  |  |  |     in_quotes = false; | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     for (p = *s; *p != '\0'; ) { | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |         int c = *p++; | 
					
						
							|  |  |  |  |         if (c == '\\') { | 
					
						
							|  |  |  |  |             if (in_quotes) { | 
					
						
							|  |  |  |  |                 ds_put_char(&out, '\\'); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (!*p) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |                 error = xasprintf("%s: backslash at end of argument", *s); | 
					
						
							|  |  |  |  |                 goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |             ds_put_char(&out, *p++); | 
					
						
							|  |  |  |  |         } else if (!in_quotes && ovsdb_token_is_delim(c)) { | 
					
						
							|  |  |  |  |             p--; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             ds_put_char(&out, c); | 
					
						
							|  |  |  |  |             if (c == '"') { | 
					
						
							|  |  |  |  |                 in_quotes = !in_quotes; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if (in_quotes) { | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |         error = xasprintf("%s: quoted string extends past end of argument", | 
					
						
							|  |  |  |  |                           *s); | 
					
						
							|  |  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     *outp = ds_cstr(&out); | 
					
						
							| 
									
										
										
										
											2010-01-27 11:25:20 -08:00
										 |  |  |  |     *s = p; | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | error: | 
					
						
							|  |  |  |  |     ds_destroy(&out); | 
					
						
							|  |  |  |  |     *outp = NULL; | 
					
						
							|  |  |  |  |     return error; | 
					
						
							| 
									
										
										
										
											2010-01-25 10:16:52 -08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Returns true if 'c' delimits tokens, or if 'c' is 0, and false otherwise. */ | 
					
						
							|  |  |  |  | bool | 
					
						
							|  |  |  |  | ovsdb_token_is_delim(unsigned char c) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     return strchr(":=, []{}", c) != NULL; | 
					
						
							|  |  |  |  | } |