Add database locking.
git-svn-id: svn://svn.h5l.se/heimdal/trunk/heimdal@2670 ec53bebd-3082-4978-b11e-865c3cabbd6b
This commit is contained in:
102
lib/hdb/db.c
102
lib/hdb/db.c
@@ -42,7 +42,7 @@ RCSID("$Id$");
|
|||||||
|
|
||||||
#ifdef HAVE_DB_H
|
#ifdef HAVE_DB_H
|
||||||
|
|
||||||
krb5_error_code
|
static krb5_error_code
|
||||||
DB_close(krb5_context context, HDB *db)
|
DB_close(krb5_context context, HDB *db)
|
||||||
{
|
{
|
||||||
DB *d = (DB*)db->db;
|
DB *d = (DB*)db->db;
|
||||||
@@ -51,45 +51,72 @@ DB_close(krb5_context context, HDB *db)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static krb5_error_code
|
||||||
|
DB_lock(krb5_context context, HDB *db, int operation)
|
||||||
|
{
|
||||||
|
DB *d = (DB*)db->db;
|
||||||
|
int fd = (*d->fd)(d);
|
||||||
|
if(fd < 0)
|
||||||
|
return HDB_ERR_CANT_LOCK_DB;
|
||||||
|
return hdb_lock(fd, operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
static krb5_error_code
|
||||||
|
DB_unlock(krb5_context context, HDB *db)
|
||||||
|
{
|
||||||
|
DB *d = (DB*)db->db;
|
||||||
|
int fd = (*d->fd)(d);
|
||||||
|
if(fd < 0)
|
||||||
|
return HDB_ERR_CANT_LOCK_DB;
|
||||||
|
return hdb_unlock(fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static krb5_error_code
|
static krb5_error_code
|
||||||
DB_op(krb5_context context, HDB *db, hdb_entry *entry, int op)
|
DB_op(krb5_context context, HDB *db, hdb_entry *entry, int op)
|
||||||
{
|
{
|
||||||
DB *d = (DB*)db->db;
|
DB *d = (DB*)db->db;
|
||||||
DBT key, value;
|
DBT key, value;
|
||||||
krb5_data data;
|
krb5_data data;
|
||||||
int err;
|
int code;
|
||||||
|
|
||||||
hdb_principal2key(context, entry->principal, &data);
|
hdb_principal2key(context, entry->principal, &data);
|
||||||
key.data = data.data;
|
key.data = data.data;
|
||||||
key.size = data.length;
|
key.size = data.length;
|
||||||
switch(op){
|
switch(op){
|
||||||
case 0:
|
case 0:
|
||||||
#if 0
|
code = db->lock(context, db, HDB_RLOCK);
|
||||||
krb5_free_principal (context, entry->principal);
|
if(code)
|
||||||
#endif
|
return code;
|
||||||
err = d->get(d, &key, &value, 0);
|
code = d->get(d, &key, &value, 0);
|
||||||
|
db->unlock(context, db); /* XXX check value */
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
hdb_entry2value(context, entry, &data);
|
hdb_entry2value(context, entry, &data);
|
||||||
value.data = data.data;
|
value.data = data.data;
|
||||||
value.size = data.length;
|
value.size = data.length;
|
||||||
err = d->put(d, &key, &value, 0);
|
code = db->lock(context, db, HDB_WLOCK);
|
||||||
|
if(code)
|
||||||
|
return code;
|
||||||
|
code = d->put(d, &key, &value, 0);
|
||||||
|
db->unlock(context, db); /* XXX check value */
|
||||||
krb5_data_free(&data);
|
krb5_data_free(&data);
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
#if 0
|
code = db->lock(context, db, HDB_WLOCK);
|
||||||
krb5_free_principal (context, entry->principal);
|
if(code)
|
||||||
#endif
|
return code;
|
||||||
err = d->del(d, &key, 0);
|
code = d->del(d, &key, 0);
|
||||||
|
db->unlock(context, db); /* XXX check value */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
data.data = key.data;
|
data.data = key.data;
|
||||||
data.length = key.size;
|
data.length = key.size;
|
||||||
krb5_data_free(&data);
|
krb5_data_free(&data);
|
||||||
if(err < 0)
|
if(code < 0)
|
||||||
return errno;
|
return errno;
|
||||||
if(err == 1)
|
if(code == 1)
|
||||||
return KRB5_HDB_NOENTRY;
|
return HDB_ERR_NOENTRY;
|
||||||
if(op == 0){
|
if(op == 0){
|
||||||
data.data = value.data;
|
data.data = value.data;
|
||||||
data.length = value.size;
|
data.length = value.size;
|
||||||
@@ -123,14 +150,18 @@ DB_seq(krb5_context context, HDB *db, hdb_entry *entry, int flag)
|
|||||||
DB *d = (DB*)db->db;
|
DB *d = (DB*)db->db;
|
||||||
DBT key, value;
|
DBT key, value;
|
||||||
krb5_data key_data, data;
|
krb5_data key_data, data;
|
||||||
int err;
|
int code;
|
||||||
krb5_principal principal;
|
krb5_principal principal;
|
||||||
|
|
||||||
err = d->seq(d, &key, &value, flag);
|
code = db->lock(context, db, HDB_RLOCK);
|
||||||
if(err == -1)
|
if(code == -1)
|
||||||
|
return HDB_ERR_DB_INUSE;
|
||||||
|
code = d->seq(d, &key, &value, flag);
|
||||||
|
db->unlock(context, db); /* XXX check value */
|
||||||
|
if(code == -1)
|
||||||
return errno;
|
return errno;
|
||||||
if(err == 1)
|
if(code == 1)
|
||||||
return KRB5_HDB_NOENTRY;
|
return HDB_ERR_NOENTRY;
|
||||||
|
|
||||||
key_data.data = key.data;
|
key_data.data = key.data;
|
||||||
key_data.length = key.size;
|
key_data.length = key.size;
|
||||||
@@ -139,7 +170,7 @@ DB_seq(krb5_context context, HDB *db, hdb_entry *entry, int flag)
|
|||||||
hdb_value2entry(context, &data, entry);
|
hdb_value2entry(context, &data, entry);
|
||||||
if (entry->principal == NULL) {
|
if (entry->principal == NULL) {
|
||||||
entry->principal = malloc(sizeof(*entry->principal));
|
entry->principal = malloc(sizeof(*entry->principal));
|
||||||
hdb_key2principal(context, &key_data, &entry->principal);
|
hdb_key2principal(context, &key_data, entry->principal);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -158,13 +189,38 @@ DB_nextkey(krb5_context context, HDB *db, hdb_entry *entry)
|
|||||||
return DB_seq(context, db, entry, R_NEXT);
|
return DB_seq(context, db, entry, R_NEXT);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static krb5_error_code
|
||||||
|
DB__get(krb5_context context, HDB *db, krb5_data key, krb5_data *reply)
|
||||||
|
{
|
||||||
|
DB *d = (DB*)db->db;
|
||||||
|
DBT k, v;
|
||||||
|
int code;
|
||||||
|
|
||||||
|
k.data = key.data;
|
||||||
|
k.size = key.length;
|
||||||
|
code = db->lock(context, db, HDB_RLOCK);
|
||||||
|
if(code)
|
||||||
|
return code;
|
||||||
|
code = d->get(d, &k, &v, 0);
|
||||||
|
db->unlock(context, db);
|
||||||
|
if(code < 0)
|
||||||
|
return errno;
|
||||||
|
if(code == 1)
|
||||||
|
return HDB_ERR_NOENTRY;
|
||||||
|
|
||||||
|
reply->length = v.size;
|
||||||
|
reply->data = malloc(reply->length);
|
||||||
|
memcpy(reply->data, v.data, reply->length);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
krb5_error_code
|
krb5_error_code
|
||||||
hdb_db_open(krb5_context context, HDB **db,
|
hdb_db_open(krb5_context context, HDB **db,
|
||||||
const char *filename, int flags, mode_t mode)
|
const char *filename, int flags, mode_t mode)
|
||||||
{
|
{
|
||||||
DB *d;
|
DB *d;
|
||||||
char *fn = malloc(strlen(filename) + 4);
|
char *fn;
|
||||||
sprintf(fn, "%s.db", filename);
|
asprintf(&fn, "%s.db", filename);
|
||||||
d = dbopen(fn, flags, mode, DB_BTREE, NULL);
|
d = dbopen(fn, flags, mode, DB_BTREE, NULL);
|
||||||
free(fn);
|
free(fn);
|
||||||
if(d == NULL)
|
if(d == NULL)
|
||||||
@@ -177,6 +233,8 @@ hdb_db_open(krb5_context context, HDB **db,
|
|||||||
(*db)->delete = DB_delete;
|
(*db)->delete = DB_delete;
|
||||||
(*db)->firstkey = DB_firstkey;
|
(*db)->firstkey = DB_firstkey;
|
||||||
(*db)->nextkey= DB_nextkey;
|
(*db)->nextkey= DB_nextkey;
|
||||||
|
(*db)->lock = DB_lock;
|
||||||
|
(*db)->unlock = DB_unlock;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user