1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-11 05:18:09 +03:00

CVE-2023-0614 ldb: Make ldb_filter_attrs_in_place() work in place

ldb_filter_attrs() previously did too much. Now its replacement,
ldb_filter_attrs_in_place(), only does the actual filtering, while
taking ownership of each element's values is handled in a separate
function, ldb_msg_elements_take_ownership().

Also, ldb_filter_attrs_in_place() no longer adds the distinguishedName
to the message if it is missing. That is handled in another function,
ldb_msg_add_distinguished_name().

As we're now modifying the original message rather than copying it into
a new one, we no longer need the filtered_msg parameter.

We adapt a test, based on ldb_filter_attrs_test, to exercise the new
function.

BUG: https://bugzilla.samba.org/show_bug.cgi?id=15270

Signed-off-by: Joseph Sutton <josephsutton@catalyst.net.nz>
Reviewed-by: Andrew Bartlett <abartlet@samba.org>
This commit is contained in:
Joseph Sutton 2023-03-03 17:30:19 +13:00 committed by Andrew Bartlett
parent 131d417604
commit f25b1756aa
4 changed files with 307 additions and 442 deletions

View File

@ -86,7 +86,7 @@ ldb_errstring: const char *(struct ldb_context *)
ldb_extended: int (struct ldb_context *, const char *, void *, struct ldb_result **) ldb_extended: int (struct ldb_context *, const char *, void *, struct ldb_result **)
ldb_extended_default_callback: int (struct ldb_request *, struct ldb_reply *) ldb_extended_default_callback: int (struct ldb_request *, struct ldb_reply *)
ldb_filter_attrs: int (struct ldb_context *, const struct ldb_message *, const char * const *, struct ldb_message *) ldb_filter_attrs: int (struct ldb_context *, const struct ldb_message *, const char * const *, struct ldb_message *)
ldb_filter_attrs_in_place: int (struct ldb_context *, const struct ldb_message *, const char * const *, struct ldb_message *) ldb_filter_attrs_in_place: int (struct ldb_message *, const char * const *)
ldb_filter_from_tree: char *(TALLOC_CTX *, const struct ldb_parse_tree *) ldb_filter_from_tree: char *(TALLOC_CTX *, const struct ldb_parse_tree *)
ldb_get_config_basedn: struct ldb_dn *(struct ldb_context *) ldb_get_config_basedn: struct ldb_dn *(struct ldb_context *)
ldb_get_create_perms: unsigned int (struct ldb_context *) ldb_get_create_perms: unsigned int (struct ldb_context *)

View File

@ -1264,19 +1264,16 @@ failed:
/* /*
* filter the specified list of attributes from msg, * filter the specified list of attributes from msg,
* adding requested attributes, and perhaps all for *, * adding requested attributes, and perhaps all for *.
* but not the DN to filtered_msg. * Unlike ldb_filter_attrs(), the DN will not be added
* if it is missing.
*/ */
int ldb_filter_attrs_in_place(struct ldb_context *ldb, int ldb_filter_attrs_in_place(struct ldb_message *msg,
const struct ldb_message *msg, const char *const *attrs)
const char *const *attrs,
struct ldb_message *filtered_msg)
{ {
unsigned int i; unsigned int i = 0;
bool keep_all = false; bool keep_all = false;
bool add_dn = false; unsigned int num_del = 0;
uint32_t num_elements;
uint32_t elements_size;
if (attrs) { if (attrs) {
/* check for special attrs */ /* check for special attrs */
@ -1286,123 +1283,41 @@ int ldb_filter_attrs_in_place(struct ldb_context *ldb,
keep_all = true; keep_all = true;
break; break;
} }
cmp = ldb_attr_cmp(attrs[i], "distinguishedName");
if (cmp == 0) {
add_dn = true;
} }
if (!keep_all && i == 0) {
msg->num_elements = 0;
return LDB_SUCCESS;
} }
} else { } else {
keep_all = true; keep_all = true;
} }
if (keep_all) {
add_dn = true;
elements_size = msg->num_elements + 1;
/* Shortcuts for the simple cases */
} else if (add_dn && i == 1) {
if (ldb_msg_add_distinguished_name(filtered_msg) != 0) {
goto failed;
}
return 0;
} else if (i == 0) {
return 0;
/*
* Otherwise we are copying at most as many elements as we
* have attributes
*/
} else {
elements_size = i;
}
filtered_msg->elements = talloc_array(filtered_msg,
struct ldb_message_element,
elements_size);
if (filtered_msg->elements == NULL) goto failed;
num_elements = 0;
for (i = 0; i < msg->num_elements; i++) { for (i = 0; i < msg->num_elements; i++) {
struct ldb_message_element *el = &msg->elements[i]; bool found = false;
/*
* el2 is assigned after the Pigeonhole principle
* check below for clarity
*/
struct ldb_message_element *el2 = NULL;
unsigned int j; unsigned int j;
if (keep_all == false) { if (keep_all) {
bool found = false; found = true;
} else {
for (j = 0; attrs[j]; j++) { for (j = 0; attrs[j]; j++) {
int cmp = ldb_attr_cmp(el->name, attrs[j]); int cmp = ldb_attr_cmp(msg->elements[i].name, attrs[j]);
if (cmp == 0) { if (cmp == 0) {
found = true; found = true;
break; break;
} }
} }
if (found == false) { }
continue;
if (!found) {
++num_del;
} else if (num_del != 0) {
msg->elements[i - num_del] = msg->elements[i];
} }
} }
/* msg->num_elements -= num_del;
* Pigeonhole principle: we can't have more elements
* than the number of attributes if they are unique in
* the DB.
*/
if (num_elements >= elements_size) {
goto failed;
}
el2 = &filtered_msg->elements[num_elements]; return LDB_SUCCESS;
*el2 = *el;
el2->name = talloc_strdup(filtered_msg->elements,
el->name);
if (el2->name == NULL) {
goto failed;
}
el2->values = talloc_array(filtered_msg->elements,
struct ldb_val, el->num_values);
if (el2->values == NULL) {
goto failed;
}
for (j=0;j<el->num_values;j++) {
el2->values[j] = ldb_val_dup(el2->values, &el->values[j]);
if (el2->values[j].data == NULL && el->values[j].length != 0) {
goto failed;
}
}
num_elements++;
}
filtered_msg->num_elements = num_elements;
if (add_dn) {
if (ldb_msg_add_distinguished_name(filtered_msg) != 0) {
goto failed;
}
}
if (filtered_msg->num_elements > 0) {
filtered_msg->elements
= talloc_realloc(filtered_msg,
filtered_msg->elements,
struct ldb_message_element,
filtered_msg->num_elements);
if (filtered_msg->elements == NULL) {
goto failed;
}
} else {
TALLOC_FREE(filtered_msg->elements);
}
return 0;
failed:
TALLOC_FREE(filtered_msg->elements);
return -1;
} }
/* Have an unpacked ldb message take talloc ownership of its elements. */ /* Have an unpacked ldb message take talloc ownership of its elements. */

View File

@ -545,13 +545,12 @@ int ldb_filter_attrs(struct ldb_context *ldb,
/* /*
* filter the specified list of attributes from msg, * filter the specified list of attributes from msg,
* adding requested attributes, and perhaps all for *, * adding requested attributes, and perhaps all for *.
* but not the DN to filtered_msg. * Unlike ldb_filter_attrs(), the DN will not be added
* if it is missing.
*/ */
int ldb_filter_attrs_in_place(struct ldb_context *ldb, int ldb_filter_attrs_in_place(struct ldb_message *msg,
const struct ldb_message *msg, const char *const *attrs);
const char *const *attrs,
struct ldb_message *filtered_msg);
/* Have an unpacked ldb message take talloc ownership of its elements. */ /* Have an unpacked ldb message take talloc ownership of its elements. */
int ldb_msg_elements_take_ownership(struct ldb_message *msg); int ldb_msg_elements_take_ownership(struct ldb_message *msg);

File diff suppressed because it is too large Load Diff