1
0
mirror of https://github.com/samba-team/samba.git synced 2024-12-25 23:21:54 +03:00
samba-mirror/source4/dsdb/schema/schema_prefixmap.c
Kamen Mazdrashki 3fe4310df7 s4/drs: Implement binary-oid-lookup into prefixMap
Signed-off-by: Stefan Metzmacher <metze@samba.org>
2009-11-06 14:05:38 +01:00

307 lines
7.9 KiB
C

/*
Unix SMB/CIFS implementation.
DRS::prefixMap implementation
Copyright (C) Kamen Mazdrashki <kamen.mazdrashki@postpath.com> 2009
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "includes.h"
#include "dsdb/samdb/samdb.h"
#include "librpc/gen_ndr/ndr_drsuapi.h"
#include "librpc/gen_ndr/ndr_drsblobs.h"
#include "../lib/util/asn1.h"
/**
* Initial prefixMap creation according to:
* [MS-DRSR] section 5.12.2
*/
WERROR dsdb_schema_pfm_new(TALLOC_CTX *mem_ctx, struct dsdb_schema_prefixmap **ppfm)
{
uint32_t i;
struct dsdb_schema_prefixmap *pfm;
const struct {
uint32_t id;
const char *oid_prefix;
} pfm_init_data[] = {
{.id=0x00000000, .oid_prefix="2.5.4"},
{.id=0x00000001, .oid_prefix="2.5.6"},
{.id=0x00000002, .oid_prefix="1.2.840.113556.1.2"},
{.id=0x00000003, .oid_prefix="1.2.840.113556.1.3"},
{.id=0x00000004, .oid_prefix="2.16.840.1.101.2.2.1"},
{.id=0x00000005, .oid_prefix="2.16.840.1.101.2.2.3"},
{.id=0x00000006, .oid_prefix="2.16.840.1.101.2.1.5"},
{.id=0x00000007, .oid_prefix="2.16.840.1.101.2.1.4"},
{.id=0x00000008, .oid_prefix="2.5.5"},
{.id=0x00000009, .oid_prefix="1.2.840.113556.1.4"},
{.id=0x0000000A, .oid_prefix="1.2.840.113556.1.5"},
{.id=0x00000013, .oid_prefix="0.9.2342.19200300.100"},
{.id=0x00000014, .oid_prefix="2.16.840.1.113730.3"},
{.id=0x00000015, .oid_prefix="0.9.2342.19200300.100.1"},
{.id=0x00000016, .oid_prefix="2.16.840.1.113730.3.1"},
{.id=0x00000017, .oid_prefix="1.2.840.113556.1.5.7000"},
{.id=0x00000018, .oid_prefix="2.5.21"},
{.id=0x00000019, .oid_prefix="2.5.18"},
{.id=0x0000001A, .oid_prefix="2.5.20"},
};
/* allocate mem for prefix map */
pfm = talloc_zero(mem_ctx, struct dsdb_schema_prefixmap);
W_ERROR_HAVE_NO_MEMORY(pfm);
pfm->length = ARRAY_SIZE(pfm_init_data);
pfm->prefixes = talloc_array(pfm, struct dsdb_schema_prefixmap_oid, pfm->length);
W_ERROR_HAVE_NO_MEMORY(pfm->prefixes);
/* build prefixes */
for (i = 0; i < pfm->length; i++) {
if (!ber_write_partial_OID_String(pfm, &pfm->prefixes[i].bin_oid, pfm_init_data[i].oid_prefix)) {
talloc_free(pfm);
return WERR_INTERNAL_ERROR;
}
pfm->prefixes[i].id = pfm_init_data[i].id;
}
*ppfm = pfm;
return WERR_OK;
}
/**
* Adds oid to prefix map.
* On success returns ID for newly added index
* or ID of existing entry that matches oid
* Reference: [MS-DRSR] section 5.12.2
*
* \param pfm prefixMap
* \param bin_oid OID prefix to be added to prefixMap
* \param pfm_id Location where to store prefixMap entry ID
*/
static WERROR _dsdb_schema_pfm_add_entry(struct dsdb_schema_prefixmap *pfm, DATA_BLOB bin_oid, uint32_t *_idx)
{
uint32_t i;
struct dsdb_schema_prefixmap_oid * pfm_entry;
struct dsdb_schema_prefixmap_oid * prefixes_new;
/* dup memory for bin-oid prefix to be added */
bin_oid = data_blob_dup_talloc(pfm, &bin_oid);
if (!bin_oid.data) {
return WERR_NOMEM;
}
/* make room for new entry */
prefixes_new = talloc_realloc(pfm, pfm->prefixes, struct dsdb_schema_prefixmap_oid, pfm->length + 1);
if (!prefixes_new) {
talloc_free(bin_oid.data);
return WERR_NOMEM;
}
pfm->prefixes = prefixes_new;
/* make new unique ID in prefixMap */
pfm_entry = &pfm->prefixes[pfm->length];
pfm_entry->id = 0;
for (i = 0; i < pfm->length; i++) {
if (pfm_entry->id < pfm->prefixes[i].id)
pfm_entry->id = pfm->prefixes[i].id;
}
/* add new bin-oid prefix */
pfm_entry->id++;
pfm_entry->bin_oid = bin_oid;
*_idx = pfm->length;
pfm->length++;
return WERR_OK;
}
/**
* Make partial binary OID for supplied OID.
* Reference: [MS-DRSR] section 5.12.2
*/
static WERROR _dsdb_pfm_make_binary_oid(const char *full_oid, TALLOC_CTX *mem_ctx,
DATA_BLOB *_bin_oid, uint32_t *_last_subid)
{
uint32_t last_subid;
const char *oid_subid;
/* make last sub-identifier value */
oid_subid = strrchr(full_oid, '.');
if (!oid_subid) {
return WERR_INVALID_PARAMETER;
}
oid_subid++;
last_subid = strtoul(oid_subid, NULL, 10);
/* encode oid in BER format */
if (!ber_write_OID_String(mem_ctx, _bin_oid, full_oid)) {
return WERR_INTERNAL_ERROR;
}
/* get the prefix of the OID */
if (last_subid < 128) {
_bin_oid->length -= 1;
} else {
_bin_oid->length -= 2;
}
/* return last_value if requested */
if (_last_subid) {
*_last_subid = last_subid;
}
return WERR_OK;
}
/**
* Lookup partial-binary-oid in prefixMap
*/
WERROR dsdb_schema_pfm_find_binary_oid(struct dsdb_schema_prefixmap *pfm,
DATA_BLOB bin_oid,
uint32_t *_idx)
{
uint32_t i;
for (i = 0; i < pfm->length; i++) {
if (pfm->prefixes[i].bin_oid.length != bin_oid.length) {
continue;
}
if (memcmp(pfm->prefixes[i].bin_oid.data, bin_oid.data, bin_oid.length) == 0) {
if (_idx) {
*_idx = i;
}
return WERR_OK;
}
}
return WERR_DS_NO_MSDS_INTID;
}
/**
* Make ATTID for given OID
* Reference: [MS-DRSR] section 5.12.2
*/
WERROR dsdb_schema_pfm_make_attid(struct dsdb_schema_prefixmap *pfm, const char *oid, uint32_t *attid)
{
WERROR werr;
uint32_t idx;
uint32_t lo_word, hi_word;
uint32_t last_subid;
DATA_BLOB bin_oid;
if (!pfm) {
return WERR_INVALID_PARAMETER;
}
if (!oid) {
return WERR_INVALID_PARAMETER;
}
werr = _dsdb_pfm_make_binary_oid(oid, pfm, &bin_oid, &last_subid);
W_ERROR_NOT_OK_RETURN(werr);
/* search the prefix in the prefix table, if none found, add
* one entry for new prefix.
*/
werr = dsdb_schema_pfm_find_binary_oid(pfm, bin_oid, &idx);
if (W_ERROR_IS_OK(werr)) {
/* free memory allocated for bin_oid */
data_blob_free(&bin_oid);
} else {
/* entry does not exists, add it */
werr = _dsdb_schema_pfm_add_entry(pfm, bin_oid, &idx);
W_ERROR_NOT_OK_RETURN(werr);
}
/* compose the attid */
lo_word = last_subid % 16384; /* actually get lower 14 bits: lo_word & 0x3FFF */
if (last_subid >= 16384) {
/* mark it so that it is known to not be the whole lastValue
* This will raise 16-th bit*/
lo_word += 32768;
}
hi_word = pfm->prefixes[idx].id;
/* make ATTID:
* HIWORD is prefixMap id
* LOWORD is truncated binary-oid */
*attid = (hi_word * 65536) + lo_word;
return WERR_OK;
}
/**
* Make OID for given ATTID.
* Reference: [MS-DRSR] section 5.12.2
*/
WERROR dsdb_schema_pfm_oid_from_attid(struct dsdb_schema_prefixmap *pfm, uint32_t attid,
TALLOC_CTX *mem_ctx, const char **_oid)
{
int i;
uint32_t hi_word, lo_word;
DATA_BLOB bin_oid = {NULL, 0};
struct dsdb_schema_prefixmap_oid *pfm_entry;
WERROR werr = WERR_OK;
/* crack attid value */
hi_word = attid >> 16;
lo_word = attid & 0xFFFF;
/* locate corRespoNding prefixMap entry */
pfm_entry = NULL;
for (i = 0; i < pfm->length; i++) {
if (hi_word == pfm->prefixes[i].id) {
pfm_entry = &pfm->prefixes[i];
break;
}
}
if (!pfm_entry) {
return WERR_INTERNAL_ERROR;
}
/* copy oid prefix making enough room */
bin_oid.length = pfm_entry->bin_oid.length + 2;
bin_oid.data = talloc_array(mem_ctx, uint8_t, bin_oid.length);
W_ERROR_HAVE_NO_MEMORY(bin_oid.data);
memcpy(bin_oid.data, pfm_entry->bin_oid.data, pfm_entry->bin_oid.length);
if (lo_word < 128) {
bin_oid.length = bin_oid.length - 1;
bin_oid.data[bin_oid.length-1] = lo_word;
}
else {
if (lo_word >= 32768) {
lo_word -= 32768;
}
bin_oid.data[bin_oid.length-2] = (0x80 | ((lo_word>>7) & 0x7f));
bin_oid.data[bin_oid.length-1] = lo_word & 0x7f;
}
if (!ber_read_OID_String(mem_ctx, bin_oid, _oid)) {
werr = WERR_INTERNAL_ERROR;
}
/* free locally allocated memory */
talloc_free(bin_oid.data);
return werr;
}