blob: e194ee01c8ca46949a990f9f187291a9f146c0a1 [file] [log] [blame]
/* Copyright (C) 2017 Mellanox Technologies Inc. */
struct semanage_ibpkey;
struct semanage_ibpkey_key;
typedef struct semanage_ibpkey_key record_key_t;
typedef struct semanage_ibpkey record_t;
#define DBASE_RECORD_DEFINED
#include <stdlib.h>
#include <string.h>
#include <netinet/in.h>
#include "ibpkey_internal.h"
#include "debug.h"
#include "handle.h"
#include "database.h"
int semanage_ibpkey_modify_local(semanage_handle_t *handle,
const semanage_ibpkey_key_t *key,
const semanage_ibpkey_t *data)
{
dbase_config_t *dconfig = semanage_ibpkey_dbase_local(handle);
return dbase_modify(handle, dconfig, key, data);
}
int semanage_ibpkey_del_local(semanage_handle_t *handle,
const semanage_ibpkey_key_t *key)
{
dbase_config_t *dconfig = semanage_ibpkey_dbase_local(handle);
return dbase_del(handle, dconfig, key);
}
int semanage_ibpkey_query_local(semanage_handle_t *handle,
const semanage_ibpkey_key_t *key,
semanage_ibpkey_t **response)
{
dbase_config_t *dconfig = semanage_ibpkey_dbase_local(handle);
return dbase_query(handle, dconfig, key, response);
}
int semanage_ibpkey_exists_local(semanage_handle_t *handle,
const semanage_ibpkey_key_t *key,
int *response)
{
dbase_config_t *dconfig = semanage_ibpkey_dbase_local(handle);
return dbase_exists(handle, dconfig, key, response);
}
int semanage_ibpkey_count_local(semanage_handle_t *handle,
unsigned int *response)
{
dbase_config_t *dconfig = semanage_ibpkey_dbase_local(handle);
return dbase_count(handle, dconfig, response);
}
int semanage_ibpkey_iterate_local(semanage_handle_t *handle,
int (*handler)(const semanage_ibpkey_t *record,
void *varg), void *handler_arg)
{
dbase_config_t *dconfig = semanage_ibpkey_dbase_local(handle);
return dbase_iterate(handle, dconfig, handler, handler_arg);
}
int semanage_ibpkey_list_local(semanage_handle_t *handle,
semanage_ibpkey_t ***records, unsigned int *count)
{
dbase_config_t *dconfig = semanage_ibpkey_dbase_local(handle);
return dbase_list(handle, dconfig, records, count);
}
hidden_def(semanage_ibpkey_list_local)
int hidden semanage_ibpkey_validate_local(semanage_handle_t *handle)
{
semanage_ibpkey_t **ibpkeys = NULL;
unsigned int nibpkeys = 0;
unsigned int i = 0, j = 0;
uint64_t subnet_prefix;
uint64_t subnet_prefix2;
char *subnet_prefix_str;
char *subnet_prefix_str2;
int low, high;
int low2, high2;
/* List and sort the ibpkeys */
if (semanage_ibpkey_list_local(handle, &ibpkeys, &nibpkeys) < 0)
goto err;
qsort(ibpkeys, nibpkeys, sizeof(semanage_ibpkey_t *),
(int (*)(const void *, const void *))
&semanage_ibpkey_compare2_qsort);
/* Test each ibpkey for overlap */
while (i < nibpkeys) {
if (STATUS_SUCCESS != semanage_ibpkey_get_subnet_prefix(handle,
ibpkeys[i],
&subnet_prefix_str)) {
ERR(handle, "Couldn't get subnet prefix string");
goto err;
}
subnet_prefix = semanage_ibpkey_get_subnet_prefix_bytes(ibpkeys[i]);
low = semanage_ibpkey_get_low(ibpkeys[i]);
high = semanage_ibpkey_get_high(ibpkeys[i]);
/* Find the first ibpkey with matching
* subnet_prefix to compare against
*/
do {
if (j == nibpkeys - 1)
goto next;
j++;
if (STATUS_SUCCESS !=
semanage_ibpkey_get_subnet_prefix(handle,
ibpkeys[j],
&subnet_prefix_str2)) {
ERR(handle, "Couldn't get subnet prefix string");
goto err;
}
subnet_prefix2 = semanage_ibpkey_get_subnet_prefix_bytes(ibpkeys[j]);
low2 = semanage_ibpkey_get_low(ibpkeys[j]);
high2 = semanage_ibpkey_get_high(ibpkeys[j]);
} while (subnet_prefix != subnet_prefix2);
/* Overlap detected */
if (low2 <= high) {
ERR(handle, "ibpkey overlap between ranges "
"(%s) %u - %u <--> (%s) %u - %u.",
subnet_prefix_str, low, high,
subnet_prefix_str2, low2, high2);
goto invalid;
}
/* If closest ibpkey of matching subnet prefix doesn't overlap
* with test ibpkey, neither do the rest of them, because that's
* how the sort function works on ibpkeys - lower bound
* ibpkeys come first
*/
next:
i++;
j = i;
}
for (i = 0; i < nibpkeys; i++)
semanage_ibpkey_free(ibpkeys[i]);
free(ibpkeys);
return STATUS_SUCCESS;
err:
ERR(handle, "could not complete ibpkeys validity check");
invalid:
for (i = 0; i < nibpkeys; i++)
semanage_ibpkey_free(ibpkeys[i]);
free(ibpkeys);
return STATUS_ERR;
}