/* * virmacmap.c: MAC address <-> Domain name mapping * * Copyright (C) 2016 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . */ #include #include "virmacmap.h" #include "virobject.h" #include "virlog.h" #include "virjson.h" #include "virfile.h" #include "virhash.h" #include "virstring.h" #include "viralloc.h" #define VIR_FROM_THIS VIR_FROM_NETWORK VIR_LOG_INIT("util.virmacmap"); /** * VIR_MAC_MAP_FILE_SIZE_MAX: * * Macro providing the upper limit on the size of mac maps file */ #define VIR_MAC_MAP_FILE_SIZE_MAX (32 * 1024 * 1024) struct virMacMap { virObjectLockable parent; GHashTable *macs; }; static virClass *virMacMapClass; static void virMacMapDispose(void *obj) { virMacMap *mgr = obj; GHashTableIter htitr; void *value; g_hash_table_iter_init(&htitr, mgr->macs); while (g_hash_table_iter_next(&htitr, NULL, &value)) g_slist_free_full(value, g_free); virHashFree(mgr->macs); } static int virMacMapOnceInit(void) { if (!VIR_CLASS_NEW(virMacMap, virClassForObjectLockable())) return -1; return 0; } VIR_ONCE_GLOBAL_INIT(virMacMap); static void virMacMapAddLocked(virMacMap *mgr, const char *domain, const char *mac) { GSList *orig_list; GSList *list; GSList *next; list = orig_list = g_hash_table_lookup(mgr->macs, domain); for (next = list; next; next = next->next) { if (STREQ((const char *) next->data, mac)) return; } list = g_slist_append(list, g_strdup(mac)); if (list != orig_list) g_hash_table_insert(mgr->macs, g_strdup(domain), list); } static void virMacMapRemoveLocked(virMacMap *mgr, const char *domain, const char *mac) { GSList *orig_list; GSList *list; GSList *next; list = orig_list = g_hash_table_lookup(mgr->macs, domain); if (!orig_list) return; for (next = list; next; next = next->next) { if (STREQ((const char *) next->data, mac)) { list = g_slist_remove_link(list, next); g_slist_free_full(next, g_free); break; } } if (list != orig_list) { if (list) g_hash_table_insert(mgr->macs, g_strdup(domain), list); else g_hash_table_remove(mgr->macs, domain); } } static int virMacMapLoadFile(virMacMap *mgr, const char *file) { g_autofree char *map_str = NULL; g_autoptr(virJSONValue) map = NULL; int map_str_len = 0; size_t i; if (virFileExists(file) && (map_str_len = virFileReadAll(file, VIR_MAC_MAP_FILE_SIZE_MAX, &map_str)) < 0) return -1; if (map_str_len == 0) return 0; if (!(map = virJSONValueFromString(map_str))) { virReportError(VIR_ERR_INTERNAL_ERROR, _("invalid json in file: %s"), file); return -1; } if (!virJSONValueIsArray(map)) { virReportError(VIR_ERR_INTERNAL_ERROR, _("Malformed file structure: %s"), file); return -1; } for (i = 0; i < virJSONValueArraySize(map); i++) { virJSONValue *tmp = virJSONValueArrayGet(map, i); virJSONValue *macs; const char *domain; size_t j; GSList *vals = NULL; if (!(domain = virJSONValueObjectGetString(tmp, "domain"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Missing domain")); return -1; } if (!(macs = virJSONValueObjectGetArray(tmp, "macs"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Missing macs")); return -1; } if (g_hash_table_contains(mgr->macs, domain)) { virReportError(VIR_ERR_INTERNAL_ERROR, _("duplicate domain '%s'"), domain); return -1; } for (j = 0; j < virJSONValueArraySize(macs); j++) { virJSONValue *macJSON = virJSONValueArrayGet(macs, j); vals = g_slist_prepend(vals, g_strdup(virJSONValueGetString(macJSON))); } vals = g_slist_reverse(vals); g_hash_table_insert(mgr->macs, g_strdup(domain), vals); } return 0; } static int virMACMapHashDumper(void *payload, const char *name, void *data) { g_autoptr(virJSONValue) obj = virJSONValueNewObject(); g_autoptr(virJSONValue) arr = virJSONValueNewArray(); GSList *macs = payload; GSList *next; for (next = macs; next; next = next->next) { g_autoptr(virJSONValue) m = virJSONValueNewString((const char *) next->data); if (virJSONValueArrayAppend(arr, &m) < 0) return -1; } if (virJSONValueObjectAppendString(obj, "domain", name) < 0 || virJSONValueObjectAppend(obj, "macs", &arr) < 0) return -1; if (virJSONValueArrayAppend(data, &obj) < 0) return -1; return 0; } static int virMacMapDumpStrLocked(virMacMap *mgr, char **str) { g_autoptr(virJSONValue) arr = virJSONValueNewArray(); if (virHashForEachSorted(mgr->macs, virMACMapHashDumper, arr) < 0) return -1; if (!(*str = virJSONValueToString(arr, true))) return -1; return 0; } static int virMacMapWriteFileLocked(virMacMap *mgr, const char *file) { g_autofree char *str = NULL; if (virMacMapDumpStrLocked(mgr, &str) < 0) return -1; if (virFileRewriteStr(file, 0644, str) < 0) return -1; return 0; } char * virMacMapFileName(const char *dnsmasqStateDir, const char *bridge) { char *filename; filename = g_strdup_printf("%s/%s.macs", dnsmasqStateDir, bridge); return filename; } #define VIR_MAC_HASH_TABLE_SIZE 10 virMacMap * virMacMapNew(const char *file) { virMacMap *mgr; if (virMacMapInitialize() < 0) return NULL; if (!(mgr = virObjectLockableNew(virMacMapClass))) return NULL; virObjectLock(mgr); mgr->macs = virHashNew(NULL); if (file && virMacMapLoadFile(mgr, file) < 0) goto error; virObjectUnlock(mgr); return mgr; error: virObjectUnlock(mgr); virObjectUnref(mgr); return NULL; } int virMacMapAdd(virMacMap *mgr, const char *domain, const char *mac) { virObjectLock(mgr); virMacMapAddLocked(mgr, domain, mac); virObjectUnlock(mgr); return 0; } int virMacMapRemove(virMacMap *mgr, const char *domain, const char *mac) { virObjectLock(mgr); virMacMapRemoveLocked(mgr, domain, mac); virObjectUnlock(mgr); return 0; } /* note that the returned pointer may be invalidated by other APIs in this module */ GSList * virMacMapLookup(virMacMap *mgr, const char *domain) { GSList *ret; virObjectLock(mgr); ret = virHashLookup(mgr->macs, domain); virObjectUnlock(mgr); return ret; } int virMacMapWriteFile(virMacMap *mgr, const char *filename) { int ret; virObjectLock(mgr); ret = virMacMapWriteFileLocked(mgr, filename); virObjectUnlock(mgr); return ret; } int virMacMapDumpStr(virMacMap *mgr, char **str) { int ret; virObjectLock(mgr); ret = virMacMapDumpStrLocked(mgr, str); virObjectUnlock(mgr); return ret; }