mkndx, adb: fix index searching

Additioal logic is needed to search objects on array: the object
comparer needs separate modes to match index, template or exact
object template. This should fix mkndx to be able to use old index.

fixes #10828
cute-signatures
Timo Teräs 2022-03-21 12:48:16 +02:00
parent 950972a56b
commit c6b9297bcb
4 changed files with 97 additions and 78 deletions

122
src/adb.c
View File

@ -472,72 +472,109 @@ struct adb_obj *adb_ro_obj(const struct adb_obj *o, unsigned i, struct adb_obj *
return adb_r_obj(o->db, adb_ro_val(o, i), no, schema); return adb_r_obj(o->db, adb_ro_val(o, i), no, schema);
} }
int adb_ro_cmp(const struct adb_obj *o1, const struct adb_obj *o2, unsigned i) int adb_ro_cmpobj(const struct adb_obj *tmpl, const struct adb_obj *obj, unsigned mode)
{ {
assert(o1->schema->kind == ADB_KIND_OBJECT); const struct adb_object_schema *schema = obj->schema;
assert(o1->schema == o2->schema); int is_set, r = 0;
assert(i > 0 && i < o1->schema->num_fields);
switch (*o1->schema->fields[i-1].kind) { assert(schema->kind == ADB_KIND_OBJECT);
assert(schema == tmpl->schema);
for (unsigned int i = ADBI_FIRST; i <= adb_ro_num(tmpl); i++) {
is_set = adb_ro_val(tmpl, i) != ADB_VAL_NULL;
if (mode == ADB_OBJCMP_EXACT || is_set) {
r = adb_ro_cmp(tmpl, obj, i, mode);
if (r) return r;
}
if (mode == ADB_OBJCMP_INDEX && !is_set)
return 0;
if (mode != ADB_OBJCMP_EXACT && i >= schema->num_compare)
return 0;
}
return 0;
}
int adb_ro_cmp(const struct adb_obj *tmpl, const struct adb_obj *obj, unsigned i, unsigned mode)
{
const struct adb_object_schema *schema = obj->schema;
assert(schema->kind == ADB_KIND_OBJECT);
assert(schema == tmpl->schema);
assert(i > 0 && i < schema->num_fields);
switch (*schema->fields[i-1].kind) {
case ADB_KIND_BLOB: case ADB_KIND_BLOB:
case ADB_KIND_INT: case ADB_KIND_INT:
return container_of(o1->schema->fields[i-1].kind, struct adb_scalar_schema, kind)->compare( return container_of(schema->fields[i-1].kind, struct adb_scalar_schema, kind)->compare(
o1->db, adb_ro_val(o1, i), tmpl->db, adb_ro_val(tmpl, i),
o2->db, adb_ro_val(o2, i)); obj->db, adb_ro_val(obj, i));
case ADB_KIND_OBJECT: { case ADB_KIND_OBJECT: {
struct adb_obj so1, so2; struct adb_obj stmpl, sobj;
adb_ro_obj(o1, i, &so1); adb_ro_obj(tmpl, i, &stmpl);
adb_ro_obj(o2, i, &so2); adb_ro_obj(obj, i, &sobj);
return so1.schema->compare(&so1, &so2); return adb_ro_cmpobj(&stmpl, &sobj, mode);
} }
} }
assert(0); assert(0);
} }
static struct adb *__db1, *__db2; static struct wacmp_param {
static const struct adb_object_schema *__schema; struct adb *db1, *db2;
const struct adb_object_schema *schema;
int mode;
} __wacmp_param;
static int wacmp(const void *p1, const void *p2) static int wacmp(const void *p1, const void *p2)
{ {
struct wacmp_param *wp = &__wacmp_param;
struct adb_obj o1, o2; struct adb_obj o1, o2;
adb_r_obj(__db1, *(adb_val_t *)p1, &o1, __schema); adb_r_obj(wp->db1, *(adb_val_t *)p1, &o1, wp->schema);
adb_r_obj(__db2, *(adb_val_t *)p2, &o2, __schema); adb_r_obj(wp->db2, *(adb_val_t *)p2, &o2, wp->schema);
return o1.schema->compare(&o1, &o2); return adb_ro_cmpobj(&o1, &o2, wp->mode);
} }
static int wadbcmp(const void *p1, const void *p2) static int wadbcmp(const void *p1, const void *p2)
{ {
struct wacmp_param *wp = &__wacmp_param;
struct adb a1, a2; struct adb a1, a2;
struct adb_obj o1, o2; struct adb_obj o1, o2;
adb_m_blob(&a1, adb_r_blob(__db1, *(adb_val_t *)p1), 0); adb_m_blob(&a1, adb_r_blob(wp->db1, *(adb_val_t *)p1), 0);
adb_m_blob(&a2, adb_r_blob(__db2, *(adb_val_t *)p2), 0); adb_m_blob(&a2, adb_r_blob(wp->db2, *(adb_val_t *)p2), 0);
adb_r_rootobj(&a1, &o1, __schema); adb_r_rootobj(&a1, &o1, wp->schema);
adb_r_rootobj(&a2, &o2, __schema); adb_r_rootobj(&a2, &o2, wp->schema);
return __schema->compare(&o1, &o2); return adb_ro_cmpobj(&o1, &o2, wp->mode);
} }
int adb_ra_find(struct adb_obj *arr, int cur, struct adb *db, adb_val_t val) int adb_ra_find(struct adb_obj *arr, int cur, struct adb_obj *tmpl)
{ {
adb_val_t *ndx; const struct adb_object_schema *schema = arr->schema, *item_schema;
struct adb_obj obj;
__db1 = db; assert(schema->kind == ADB_KIND_ARRAY);
__db2 = arr->db; assert(*schema->fields[0].kind == ADB_KIND_OBJECT);
__schema = arr->schema; item_schema = container_of(schema->fields[0].kind, struct adb_object_schema, kind),
assert(__schema->kind == ADB_KIND_ARRAY); assert(item_schema == tmpl->schema);
__schema = container_of(__schema->fields[0].kind, struct adb_object_schema, kind);
if (cur == 0) { if (cur == 0) {
ndx = bsearch(&val, &arr->obj[ADBI_FIRST], adb_ra_num(arr), sizeof(arr->obj[0]), wacmp); unsigned m, l = ADBI_FIRST, r = adb_ra_num(arr) + 1;
if (!ndx) return -1; while (l < r) {
cur = ndx - arr->obj; m = (l + r) / 2;
while (cur > 1 && wacmp(&val, &arr->obj[cur-1]) == 0) cur--; if (adb_ro_cmpobj(tmpl, adb_ro_obj(arr, m, &obj), ADB_OBJCMP_INDEX) < 0)
r = m;
else
l = m + 1;
}
cur = r - 1;
} else { } else {
cur++; cur++;
if (wacmp(&val, &arr->obj[cur]) != 0)
return -1;
} }
return cur;
for (; cur <= adb_ra_num(arr); cur++) {
adb_ro_obj(arr, cur, &obj);
if (adb_ro_cmpobj(tmpl, &obj, ADB_OBJCMP_TEMPLATE) == 0) return cur;
if (adb_ro_cmpobj(tmpl, &obj, ADB_OBJCMP_INDEX) != 0) return -1;
}
return -1;
} }
/* Write interface */ /* Write interface */
@ -939,15 +976,20 @@ adb_val_t adb_wa_append_fromstring(struct adb_obj *o, apk_blob_t b)
void adb_wa_sort(struct adb_obj *arr) void adb_wa_sort(struct adb_obj *arr)
{ {
assert(arr->schema->kind == ADB_KIND_ARRAY); const struct adb_object_schema *schema = arr->schema;
__db1 = __db2 = arr->db; assert(schema->kind == ADB_KIND_ARRAY);
__wacmp_param = (struct wacmp_param) {
.db1 = arr->db,
.db2 = arr->db,
.mode = ADB_OBJCMP_EXACT,
};
switch (*arr->schema->fields[0].kind) { switch (*arr->schema->fields[0].kind) {
case ADB_KIND_OBJECT: case ADB_KIND_OBJECT:
__schema = container_of(arr->schema->fields[0].kind, struct adb_object_schema, kind); __wacmp_param.schema = container_of(arr->schema->fields[0].kind, struct adb_object_schema, kind);
qsort(&arr->obj[ADBI_FIRST], adb_ra_num(arr), sizeof(arr->obj[0]), wacmp); qsort(&arr->obj[ADBI_FIRST], adb_ra_num(arr), sizeof(arr->obj[0]), wacmp);
break; break;
case ADB_KIND_ADB: case ADB_KIND_ADB:
__schema = container_of(arr->schema->fields[0].kind, struct adb_adb_schema, kind)->schema; __wacmp_param.schema = container_of(arr->schema->fields[0].kind, struct adb_adb_schema, kind)->schema;
qsort(&arr->obj[ADBI_FIRST], adb_ra_num(arr), sizeof(arr->obj[0]), wadbcmp); qsort(&arr->obj[ADBI_FIRST], adb_ra_num(arr), sizeof(arr->obj[0]), wadbcmp);
break; break;
default: default:

View File

@ -105,13 +105,17 @@ struct adb_sign_v0 {
#define ADB_ARRAY_ITEM(_t) { { .kind = &(_t).kind } } #define ADB_ARRAY_ITEM(_t) { { .kind = &(_t).kind } }
#define ADB_FIELD(_i, _n, _t) [(_i)-1] = { .name = _n, .kind = &(_t).kind } #define ADB_FIELD(_i, _n, _t) [(_i)-1] = { .name = _n, .kind = &(_t).kind }
#define ADB_OBJCMP_EXACT 0 // match all fields
#define ADB_OBJCMP_TEMPLATE 1 // match fields set on template
#define ADB_OBJCMP_INDEX 2 // match fields until first non-set one
struct adb_object_schema { struct adb_object_schema {
uint8_t kind; uint8_t kind;
uint16_t num_fields; uint16_t num_fields;
uint16_t num_compare;
apk_blob_t (*tostring)(struct adb_obj *, char *, size_t); apk_blob_t (*tostring)(struct adb_obj *, char *, size_t);
int (*fromstring)(struct adb_obj *, apk_blob_t); int (*fromstring)(struct adb_obj *, apk_blob_t);
int (*compare)(const struct adb_obj *, const struct adb_obj *);
void (*pre_commit)(struct adb_obj *); void (*pre_commit)(struct adb_obj *);
struct { struct {
@ -192,8 +196,9 @@ adb_val_t adb_ro_val(const struct adb_obj *o, unsigned i);
uint32_t adb_ro_int(const struct adb_obj *o, unsigned i); uint32_t adb_ro_int(const struct adb_obj *o, unsigned i);
apk_blob_t adb_ro_blob(const struct adb_obj *o, unsigned i); apk_blob_t adb_ro_blob(const struct adb_obj *o, unsigned i);
struct adb_obj *adb_ro_obj(const struct adb_obj *o, unsigned i, struct adb_obj *); struct adb_obj *adb_ro_obj(const struct adb_obj *o, unsigned i, struct adb_obj *);
int adb_ro_cmp(const struct adb_obj *o1, const struct adb_obj *o2, unsigned i); int adb_ro_cmpobj(const struct adb_obj *o1, const struct adb_obj *o2, unsigned mode);
int adb_ra_find(struct adb_obj *arr, int cur, struct adb *db, adb_val_t val); int adb_ro_cmp(const struct adb_obj *o1, const struct adb_obj *o2, unsigned i, unsigned mode);
int adb_ra_find(struct adb_obj *arr, int cur, struct adb_obj *tmpl);
/* Primitive write */ /* Primitive write */
void adb_w_root(struct adb *, adb_val_t); void adb_w_root(struct adb *, adb_val_t);

View File

@ -341,17 +341,12 @@ fail:
return -APKE_DEPENDENCY_FORMAT; return -APKE_DEPENDENCY_FORMAT;
} }
static int dependency_cmp(const struct adb_obj *o1, const struct adb_obj *o2)
{
return adb_ro_cmp(o1, o2, ADBI_DEP_NAME);
}
const struct adb_object_schema schema_dependency = { const struct adb_object_schema schema_dependency = {
.kind = ADB_KIND_OBJECT, .kind = ADB_KIND_OBJECT,
.num_fields = ADBI_DEP_MAX, .num_fields = ADBI_DEP_MAX,
.num_compare = ADBI_DEP_NAME,
.tostring = dependency_tostring, .tostring = dependency_tostring,
.fromstring = dependency_fromstring, .fromstring = dependency_fromstring,
.compare = dependency_cmp,
.fields = { .fields = {
ADB_FIELD(ADBI_DEP_NAME, "name", scalar_string), ADB_FIELD(ADBI_DEP_NAME, "name", scalar_string),
ADB_FIELD(ADBI_DEP_VERSION, "version", scalar_version), ADB_FIELD(ADBI_DEP_VERSION, "version", scalar_version),
@ -383,20 +378,10 @@ const struct adb_object_schema schema_dependency_array = {
.fields = ADB_ARRAY_ITEM(schema_dependency), .fields = ADB_ARRAY_ITEM(schema_dependency),
}; };
static int pkginfo_cmp(const struct adb_obj *o1, const struct adb_obj *o2)
{
int r;
r = adb_ro_cmp(o1, o2, ADBI_PI_NAME);
if (r) return r;
r = adb_ro_cmp(o1, o2, ADBI_PI_VERSION);
if (r) return r;
return adb_ro_cmp(o1, o2, ADBI_PI_UNIQUE_ID);
}
const struct adb_object_schema schema_pkginfo = { const struct adb_object_schema schema_pkginfo = {
.kind = ADB_KIND_OBJECT, .kind = ADB_KIND_OBJECT,
.num_fields = ADBI_PI_MAX, .num_fields = ADBI_PI_MAX,
.compare = pkginfo_cmp, .num_compare = ADBI_PI_UNIQUE_ID,
.fields = { .fields = {
ADB_FIELD(ADBI_PI_NAME, "name", scalar_string), ADB_FIELD(ADBI_PI_NAME, "name", scalar_string),
ADB_FIELD(ADBI_PI_VERSION, "version", scalar_version), ADB_FIELD(ADBI_PI_VERSION, "version", scalar_version),
@ -448,15 +433,10 @@ const struct adb_object_schema schema_acl = {
}, },
}; };
static int file_cmp(const struct adb_obj *o1, const struct adb_obj *o2)
{
return adb_ro_cmp(o1, o2, ADBI_FI_NAME);
}
const struct adb_object_schema schema_file = { const struct adb_object_schema schema_file = {
.kind = ADB_KIND_OBJECT, .kind = ADB_KIND_OBJECT,
.num_fields = ADBI_FI_MAX, .num_fields = ADBI_FI_MAX,
.compare = file_cmp, .num_compare = ADBI_FI_NAME,
.fields = { .fields = {
ADB_FIELD(ADBI_FI_NAME, "name", scalar_string), ADB_FIELD(ADBI_FI_NAME, "name", scalar_string),
ADB_FIELD(ADBI_FI_ACL, "acl", schema_acl), ADB_FIELD(ADBI_FI_ACL, "acl", schema_acl),
@ -477,7 +457,7 @@ const struct adb_object_schema schema_file_array = {
const struct adb_object_schema schema_dir = { const struct adb_object_schema schema_dir = {
.kind = ADB_KIND_OBJECT, .kind = ADB_KIND_OBJECT,
.num_fields = ADBI_DI_MAX, .num_fields = ADBI_DI_MAX,
.compare = file_cmp, .num_compare = ADBI_DI_NAME,
.fields = { .fields = {
ADB_FIELD(ADBI_DI_NAME, "name", scalar_string), ADB_FIELD(ADBI_DI_NAME, "name", scalar_string),
ADB_FIELD(ADBI_DI_ACL, "acl", schema_acl), ADB_FIELD(ADBI_DI_ACL, "acl", schema_acl),
@ -506,15 +486,10 @@ const struct adb_object_schema schema_scripts = {
}, },
}; };
static int package_cmp(const struct adb_obj *o1, const struct adb_obj *o2)
{
return adb_ro_cmp(o1, o2, ADBI_PKG_PKGINFO);
}
const struct adb_object_schema schema_package = { const struct adb_object_schema schema_package = {
.kind = ADB_KIND_OBJECT, .kind = ADB_KIND_OBJECT,
.num_fields = ADBI_PKG_MAX, .num_fields = ADBI_PKG_MAX,
.compare = package_cmp, .num_compare = ADBI_PKG_PKGINFO,
.fields = { .fields = {
ADB_FIELD(ADBI_PKG_PKGINFO, "info", schema_pkginfo), ADB_FIELD(ADBI_PKG_PKGINFO, "info", schema_pkginfo),
ADB_FIELD(ADBI_PKG_PATHS, "paths", schema_dir_array), ADB_FIELD(ADBI_PKG_PATHS, "paths", schema_dir_array),

View File

@ -208,7 +208,7 @@ static int mkndx_main(void *pctx, struct apk_ctx *ac, struct apk_string_array *a
adb_w_init_tmp(&tmpdb, 200); adb_w_init_tmp(&tmpdb, 200);
adb_wo_alloca(&tmpl, &schema_pkginfo, &tmpdb); adb_wo_alloca(&tmpl, &schema_pkginfo, &tmpdb);
adb_w_init_alloca(&ctx->db, ADB_SCHEMA_INDEX, 1000); adb_w_init_alloca(&ctx->db, ADB_SCHEMA_INDEX, 8000);
adb_wo_alloca(&ndx, &schema_index, &ctx->db); adb_wo_alloca(&ndx, &schema_index, &ctx->db);
adb_wo_alloca(&ctx->pkgs, &schema_pkginfo_array, &ctx->db); adb_wo_alloca(&ctx->pkgs, &schema_pkginfo_array, &ctx->db);
adb_wo_alloca(&ctx->pkginfo, &schema_pkginfo, &ctx->db); adb_wo_alloca(&ctx->pkginfo, &schema_pkginfo, &ctx->db);
@ -240,7 +240,6 @@ static int mkndx_main(void *pctx, struct apk_ctx *ac, struct apk_string_array *a
if (index_mtime >= fi.mtime) { if (index_mtime >= fi.mtime) {
char *fname, *fend; char *fname, *fend;
apk_blob_t bname, bver; apk_blob_t bname, bver;
adb_val_t match;
int i; int i;
/* Check that it looks like a package name */ /* Check that it looks like a package name */
@ -260,15 +259,13 @@ static int mkndx_main(void *pctx, struct apk_ctx *ac, struct apk_string_array *a
adb_wo_blob(&tmpl, ADBI_PI_NAME, bname); adb_wo_blob(&tmpl, ADBI_PI_NAME, bname);
adb_wo_blob(&tmpl, ADBI_PI_VERSION, bver); adb_wo_blob(&tmpl, ADBI_PI_VERSION, bver);
adb_wo_int(&tmpl, ADBI_PI_FILE_SIZE, fi.size); adb_wo_int(&tmpl, ADBI_PI_FILE_SIZE, fi.size);
match = adb_w_obj(&tmpl);
if ((i = adb_ra_find(&opkgs, 0, &tmpdb, match)) > 0) { if ((i = adb_ra_find(&opkgs, 0, &tmpl)) > 0) {
struct adb_obj pkg; struct adb_obj pkg;
adb_ro_obj(&opkgs, i, &pkg); adb_ro_obj(&opkgs, i, &pkg);
val = adb_wa_append(&ctx->pkgs, adb_w_copy(&ctx->db, &odb, adb_ro_val(&opkgs, i))); val = adb_wa_append(&ctx->pkgs, adb_w_copy(&ctx->db, &odb, adb_ro_val(&opkgs, i)));
found = TRUE; found = TRUE;
break;
} }
} }
if (!found) { if (!found) {