summaryrefslogtreecommitdiff
path: root/source3
diff options
context:
space:
mode:
Diffstat (limited to 'source3')
-rw-r--r--source3/registry/reg_backend_db.c276
1 files changed, 0 insertions, 276 deletions
diff --git a/source3/registry/reg_backend_db.c b/source3/registry/reg_backend_db.c
index a9e06b3048..d9204a5a41 100644
--- a/source3/registry/reg_backend_db.c
+++ b/source3/registry/reg_backend_db.c
@@ -49,7 +49,6 @@ static int regdb_fetch_values_internal(struct db_context *db, const char* key,
static bool regdb_store_values_internal(struct db_context *db, const char *key,
struct regval_ctr *values);
-static NTSTATUS create_sorted_subkeys(const char *key);
static WERROR regdb_create_basekey(struct db_context *db, const char *key);
static WERROR regdb_create_subkey_internal(struct db_context *db,
const char *key,
@@ -1314,281 +1313,6 @@ done:
return ret;
}
-/*
- * regdb_key_exists() is a very frequent operation. It can be quite
- * time-consuming to fully fetch the parent's subkey list, talloc_strdup all
- * subkeys and then compare the keyname linearly to all the parent's subkeys.
- *
- * The following code tries to make this operation as efficient as possible:
- * Per registry key we create a list of subkeys that is very efficient to
- * search for existence of a subkey. Its format is:
- *
- * 4 bytes num_subkeys
- * 4*num_subkey bytes offset into the string array
- * then follows a sorted list of subkeys in uppercase
- *
- * This record is created by create_sorted_subkeys() on demand if it does not
- * exist. scan_parent_subkeys() uses regdb->parse_record to search the sorted
- * list, the parsing code and the binary search can be found in
- * parent_subkey_scanner. The code uses parse_record() to avoid a memcpy of
- * the potentially large subkey record.
- *
- * The sorted subkey record is deleted in regdb_store_keys_internal2 and
- * recreated on demand.
- */
-
-static int cmp_keynames(char **p1, char **p2)
-{
- return strcasecmp_m(*p1, *p2);
-}
-
-struct create_sorted_subkeys_context {
- const char *key;
- const char *sorted_keyname;
-};
-
-static NTSTATUS create_sorted_subkeys_action(struct db_context *db,
- void *private_data)
-{
- char **sorted_subkeys;
- struct regsubkey_ctr *ctr;
- NTSTATUS status;
- char *buf;
- char *p;
- int i;
- size_t len;
- int num_subkeys;
- struct create_sorted_subkeys_context *sorted_ctx;
-
- sorted_ctx = (struct create_sorted_subkeys_context *)private_data;
-
- /*
- * In this function, we only treat failing of the actual write to
- * the db as a real error. All preliminary errors, at a stage when
- * nothing has been written to the DB yet are treated as success
- * to be committed (as an empty transaction).
- *
- * The reason is that this (disposable) call might be nested in other
- * transactions. Doing a cancel here would destroy the possibility of
- * a transaction_commit for transactions that we might be wrapped in.
- */
-
- status = werror_to_ntstatus(regsubkey_ctr_init(talloc_tos(), &ctr));
- if (!NT_STATUS_IS_OK(status)) {
- /* don't treat this as an error */
- status = NT_STATUS_OK;
- goto done;
- }
-
- status = werror_to_ntstatus(regdb_fetch_keys_internal(db,
- sorted_ctx->key,
- ctr));
- if (!NT_STATUS_IS_OK(status)) {
- /* don't treat this as an error */
- status = NT_STATUS_OK;
- goto done;
- }
-
- num_subkeys = regsubkey_ctr_numkeys(ctr);
- sorted_subkeys = talloc_array(ctr, char *, num_subkeys);
- if (sorted_subkeys == NULL) {
- /* don't treat this as an error */
- goto done;
- }
-
- len = 4 + 4*num_subkeys;
-
- for (i = 0; i < num_subkeys; i++) {
- sorted_subkeys[i] = talloc_strdup_upper(sorted_subkeys,
- regsubkey_ctr_specific_key(ctr, i));
- if (sorted_subkeys[i] == NULL) {
- /* don't treat this as an error */
- goto done;
- }
- len += strlen(sorted_subkeys[i])+1;
- }
-
- TYPESAFE_QSORT(sorted_subkeys, num_subkeys, cmp_keynames);
-
- buf = talloc_array(ctr, char, len);
- if (buf == NULL) {
- /* don't treat this as an error */
- goto done;
- }
- p = buf + 4 + 4*num_subkeys;
-
- SIVAL(buf, 0, num_subkeys);
-
- for (i=0; i < num_subkeys; i++) {
- ptrdiff_t offset = p - buf;
- SIVAL(buf, 4 + 4*i, offset);
- strlcpy(p, sorted_subkeys[i], len-offset);
- p += strlen(sorted_subkeys[i]) + 1;
- }
-
- status = dbwrap_store_bystring(
- db, sorted_ctx->sorted_keyname, make_tdb_data((uint8_t *)buf,
- len),
- TDB_REPLACE);
-
-done:
- talloc_free(ctr);
- return status;
-}
-
-static NTSTATUS create_sorted_subkeys_internal(const char *key,
- const char *sorted_keyname)
-{
- NTSTATUS status;
- struct create_sorted_subkeys_context sorted_ctx;
-
- sorted_ctx.key = key;
- sorted_ctx.sorted_keyname = sorted_keyname;
-
- status = dbwrap_trans_do(regdb,
- create_sorted_subkeys_action,
- &sorted_ctx);
-
- return status;
-}
-
-static NTSTATUS create_sorted_subkeys(const char *key)
-{
- char *sorted_subkeys_keyname;
- NTSTATUS status;
-
- sorted_subkeys_keyname = talloc_asprintf(talloc_tos(), "%s\\%s",
- REG_SORTED_SUBKEYS_PREFIX,
- key);
- if (sorted_subkeys_keyname == NULL) {
- status = NT_STATUS_NO_MEMORY;
- goto done;
- }
-
- status = create_sorted_subkeys_internal(key, sorted_subkeys_keyname);
-
-done:
- return status;
-}
-
-struct scan_subkey_state {
- char *name;
- bool scanned;
- bool found;
-};
-
-static int parent_subkey_scanner(TDB_DATA key, TDB_DATA data,
- void *private_data)
-{
- struct scan_subkey_state *state =
- (struct scan_subkey_state *)private_data;
- uint32_t num_subkeys;
- uint32_t l, u;
-
- if (data.dsize < sizeof(uint32_t)) {
- return -1;
- }
-
- state->scanned = true;
- state->found = false;
-
- tdb_unpack(data.dptr, data.dsize, "d", &num_subkeys);
-
- l = 0;
- u = num_subkeys;
-
- while (l < u) {
- uint32_t idx = (l+u)/2;
- char *s = (char *)data.dptr + IVAL(data.dptr, 4 + 4*idx);
- int comparison = strcmp(state->name, s);
-
- if (comparison < 0) {
- u = idx;
- } else if (comparison > 0) {
- l = idx + 1;
- } else {
- state->found = true;
- return 0;
- }
- }
- return 0;
-}
-
-static bool scan_parent_subkeys(struct db_context *db, const char *parent,
- const char *name)
-{
- char *path = NULL;
- char *key = NULL;
- struct scan_subkey_state state = { 0, };
- bool result = false;
- int res;
-
- state.name = NULL;
-
- path = normalize_reg_path(talloc_tos(), parent);
- if (path == NULL) {
- goto fail;
- }
-
- key = talloc_asprintf(talloc_tos(), "%s\\%s",
- REG_SORTED_SUBKEYS_PREFIX, path);
- if (key == NULL) {
- goto fail;
- }
-
- state.name = talloc_strdup_upper(talloc_tos(), name);
- if (state.name == NULL) {
- goto fail;
- }
- state.scanned = false;
-
- res = db->parse_record(db, string_term_tdb_data(key),
- parent_subkey_scanner, &state);
-
- if (state.scanned) {
- result = state.found;
- } else {
- NTSTATUS status;
-
- res = db->transaction_start(db);
- if (res != 0) {
- DEBUG(0, ("error starting transaction\n"));
- goto fail;
- }
-
- DEBUG(2, (__location__ " WARNING: recreating the sorted "
- "subkeys cache for key '%s' from scan_parent_subkeys "
- "this should not happen (too frequently)...\n",
- path));
-
- status = create_sorted_subkeys_internal(path, key);
- if (!NT_STATUS_IS_OK(status)) {
- res = db->transaction_cancel(db);
- if (res != 0) {
- smb_panic("Failed to cancel transaction.");
- }
- goto fail;
- }
-
- res = db->parse_record(db, string_term_tdb_data(key),
- parent_subkey_scanner, &state);
- if ((res == 0) && (state.scanned)) {
- result = state.found;
- }
-
- res = db->transaction_commit(db);
- if (res != 0) {
- DEBUG(0, ("error committing transaction\n"));
- result = false;
- }
- }
-
- fail:
- TALLOC_FREE(path);
- TALLOC_FREE(state.name);
- return result;
-}
-
/**
* Check for the existence of a key.
*