From: Luiz Capitulino <lcapitulino@redhat.com>
To: Markus Armbruster <armbru@redhat.com>
Cc: aliguori@us.ibm.com, qemu-devel@nongnu.org, avi@redhat.com
Subject: Re: [Qemu-devel] [PATCH 04/29] Introduce QDict
Date: Mon, 24 Aug 2009 13:11:58 -0300 [thread overview]
Message-ID: <20090824131158.414fe394@doriath> (raw)
In-Reply-To: <877hwtjk3n.fsf@pike.pond.sub.org>
On Mon, 24 Aug 2009 17:40:44 +0200
Markus Armbruster <armbru@redhat.com> wrote:
> Luiz Capitulino <lcapitulino@redhat.com> writes:
>
> > QDict is a high-level dictionary data type that can be used to store a
> > collection of QObjects. A unique key is associated with only one
> > QObject.
> >
> > The following functions are available:
> >
> > - qdict_new() Create a new dictionary
> > - qdict_add() Add a new 'key:object' pair
> > - qdict_get() Get the QObject of a given key
> > - qdict_del() Delete a 'key:object' pair
> > - qdict_size() Return the size of the dictionary
> > - qdict_exists() Check if a given 'key' exists
> >
> > Some high-level helpers to operate on QStrings and QInts objects
> > are also provided.
> >
> > Signed-off-by: Luiz Capitulino <lcapitulino@redhat.com>
> > ---
> > Makefile | 2 +-
> > qdict.c | 311 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
> > qdict.h | 40 ++++++++
> > qobject.h | 1 +
> > 4 files changed, 353 insertions(+), 1 deletions(-)
> > create mode 100644 qdict.c
> > create mode 100644 qdict.h
> >
> > diff --git a/Makefile b/Makefile
> [...]
> > diff --git a/qdict.c b/qdict.c
> > new file mode 100644
> > index 0000000..3901c48
> > --- /dev/null
> > +++ b/qdict.c
> > @@ -0,0 +1,311 @@
> > +/*
> > + * QDict data type.
> > + *
> > + * Copyright (C) 2009 Red Hat Inc.
> > + *
> > + * Authors:
> > + * Luiz Capitulino <lcapitulino@redhat.com>
> > + *
> > + * This work is licensed under the terms of the GNU GPL, version 2. See
> > + * the COPYING file in the top-level directory.
> > + */
> > +
> > +#include "qdict.h"
> > +#include "qobject.h"
> > +#include "qemu-common.h"
> > +
> > +static const QType qdict_type;
> > +
> > +/**
> > + * qdict_new(): Create a new dictionary data-type
> > + *
> > + * Return new reference.
> > + */
> > +QDict *qdict_new(void)
> > +{
> > + QDict *qdict;
> > +
> > + qdict = qemu_mallocz(sizeof(*qdict));
> > + QOBJECT_INIT(qdict, &qdict_type);
> > +
> > + return qdict;
> > +}
> > +
> > +/**
> > + * qobject_to_qdict(): Convert a QObject into a QDict
> > + */
> > +QDict *qobject_to_qdict(const QObject *obj)
> > +{
> > + if (qobject_type(obj) != QTYPE_QDICT)
> > + return NULL;
> > +
> > + return container_of(obj, QDict, base);
> > +}
> > +
> > +/**
> > + * tdb_hash(): based on the hash agorithm from gdbm, via tdb
> > + * (from module-init-tools)
> > + */
> > +static unsigned int tdb_hash(const char *name)
> > +{
> > + unsigned value; /* Used to compute the hash value. */
> > + unsigned i; /* Used to cycle through random values. */
> > +
> > + /* Set the initial value from the key size. */
> > + for (value = 0x238F13AF * strlen(name), i=0; name[i]; i++)
> > + value = (value + (((const unsigned char *)name)[i] << (i*5 % 24)));
> > +
> > + return (1103515243 * value + 12345);
> > +}
> > +
> > +/**
> > + * alloc_entry(): allocate a new QDictEntry
> > + */
> > +static QDictEntry *alloc_entry(const char *key, QObject *value,
> > + QDictEntry *next)
> > +{
> > + QDictEntry *entry;
> > +
> > + entry = qemu_malloc(sizeof(*entry));
> > + entry->key = qemu_strdup(key);
> > + entry->value = value;
> > + entry->next = next;
> > +
> > + return entry;
> > +}
> > +
> > +/**
> > + * qdict_find(): Low-level lookup function
> > + */
> > +static void *qdict_find(const QDict *qdict,
> > + const char *key, unsigned int hash)
> > +{
> > + QDictEntry *e;
> > +
> > + for (e = qdict->table[hash]; e; e = e->next)
> > + if (!strcmp(e->key, key))
> > + return e->value;
> > + return NULL;
> > +}
> > +
> > +/**
> > + * qdict_add(): Add a new object into the dictionary
> > + *
> > + * Add the pair 'key:value' into qdict. Does nothing if 'key' already
> > + * exist.
> > + *
> > + * NOTE: this function 'steals' a reference to 'value'
> > + */
> > +void qdict_add(QDict *qdict, const char *key, QObject *value)
>
> So this steals unless KEY already exists. Callers who need to know
> whether they still own VALUE afterwards have to check whether KEY exists
> before. Hmm. What about returning whether the reference was stolen?
>
> Or just steal unconditionally.
Avi has suggested having a qdict_put() with that semantics, I'm
already working on this.
> Same for the qdict_add_FOO().
>
> Is it okay to add a null value?
No, although this is not enforced at insertion time. I'll fix
in new qdict_put().
Note that supporting this is just a matter of checking if
a certain struct member if null at destroy time, but I don't
see uses for it as we have agreed the dict will only store
QObjects.
> > +{
> > + unsigned int hash;
> > + QDictEntry *entry;
> > +
> > + hash = tdb_hash(key) % QDICT_HASH_SIZE;
> > + if (qdict_find(qdict, key, hash)) {
> > + /* Don't add again if it's already there */
> > + return;
> > + }
> > +
> > + entry = alloc_entry(key, value, qdict->table[hash]);
> > + qdict->table[hash] = entry;
> > + qdict->size++;
> > +}
> > +
> > +/**
> > + * qdict_add_qint(): Add a new QInt into the dictionary
> > + *
> > + * Add the pair 'key:qint' into qdict. Does nothing if 'key' already
> > + * exist.
> > + *
> > + * NOTE: this function 'steals' a reference to 'qi'
> > + */
> > +void qdict_add_qint(QDict *qdict, const char *key, QInt *qi)
> > +{
> > + qdict_add(qdict, key, QOBJECT(qi));
> > +}
> > +
> > +/**
> > + * qdict_add_qstring(): Add a new QString into the dictionary
> > + *
> > + * Add the pair 'key:qstring' into qdict. Does nothing if 'key' already
> > + * exist.
> > + *
> > + * NOTE: this function 'steals' a reference to 'qs'
> > + */
> > +void qdict_add_qstring(QDict *qdict, const char *key, QString *qs)
> > +{
> > + qdict_add(qdict, key, QOBJECT(qs));
> > +}
> > +
> > +/**
> > + * qdict_get(): Lookup for a given 'key'
> > + *
> > + * Return borrowed reference to QObject if 'key' exists,
> > + * NULL otherwise.
> > + */
> > +QObject *qdict_get(const QDict *qdict, const char *key)
> > +{
> > + return qdict_find(qdict, key, tdb_hash(key) % QDICT_HASH_SIZE);
> > +}
> > +
> > +/**
> > + * qdict_size(): Return the size of the dictionary
> > + */
> > +size_t qdict_size(const QDict *qdict)
> > +{
> > + return qdict->size;
> > +}
> > +
> > +/**
> > + * qdict_get_obj(): Get a QObject of a specific type.
> > + */
> > +static QObject *qdict_get_obj(const QDict *qdict, const char *key,
> > + qtype_code type)
> > +{
> > + QObject *obj;
> > +
> > + obj = qdict_get(qdict, key);
> > + assert(obj != NULL);
> > + assert(qobject_type(obj) == type);
> > +
> > + return obj;
> > +}
> > +
> > +/**
> > + * qdict_get_int(): Get an int value mapped by 'key'
> > + *
> > + * This function assumes that 'key' exists and it stores a
> > + * QInt object.
> > + */
> > +int qdict_get_int(const QDict *qdict, const char *key)
> > +{
> > + QObject *obj = qdict_get_obj(qdict, key, QTYPE_QINT);
> > + return qint_to_int(qobject_to_qint(obj));
> > +}
> > +
> > +/**
> > + * qdict_get_uint32(): Get an uint32_t value mapped by 'key'
> > + *
> > + * This function assumes that 'key' exists and it stores a
> > + * QInt object.
> > + */
> > +uint32_t qdict_get_uint32(const QDict *qdict, const char *key)
> > +{
> > + QObject *obj = qdict_get_obj(qdict, key, QTYPE_QINT);
> > + return qint_to_uint32(qobject_to_qint(obj));
> > +}
> > +
> > +/**
> > + * qdict_get_uint64(): Get an uint64_t value mapped by 'key'
> > + *
> > + * This function assumes that 'key' exists and it stores a
> > + * QInt object.
> > + */
> > +uint64_t qdict_get_uint64(const QDict *qdict, const char *key)
> > +{
> > + QObject *obj = qdict_get_obj(qdict, key, QTYPE_QINT);
> > + return qint_to_uint64(qobject_to_qint(obj));
> > +}
> > +
> > +/**
> > + * qdict_get_str(): Get a pointer to the stored string mapped
> > + * by 'key'
> > + *
> > + * return the string pointer on success, NULL if 'key' doesn't
> > + * exist.
> > + */
> > +const char *qdict_get_str(const QDict *qdict, const char *key)
> > +{
> > + QObject *obj;
> > +
> > + obj = qdict_get(qdict, key);
> > + if (!obj)
> > + return NULL;
> > +
> > + assert(qobject_type(obj) == QTYPE_QSTRING);
> > + return qstring_get_str(qobject_to_qstring(obj));
> > +}
> > +
> > +/**
> > + * qdict_exists(): Check if 'key' exists
> > + *
> > + * return 1 if 'key' exists in the dict, 0 otherwise
> > + */
> > +int qdict_exists(const QDict *qdict, const char *key)
> > +{
> > + QDictEntry *e;
> > +
> > + for (e = qdict->table[tdb_hash(key) % QDICT_HASH_SIZE]; e; e = e->next)
> > + if (!strcmp(e->key, key))
> > + return 1;
> > + return 0;
> > +}
>
> Duplicates the loop from qdict_find().
>
> What about
>
> return qdict_find(qdict, key, tdb_hash(key) % QDICT_HASH_SIZE) != NULL
>
> Doesn't work if qdict can contain (key, value) pairs with null values.
> Not a problem if qdict_find() returned e (of type QDictEntry *) instead
> of e->value (of type void *).
This code is somewhat different now and the duplication is already
gone, as I have ported QDict to use list functions from "sys-queue.h".
next prev parent reply other threads:[~2009-08-24 16:12 UTC|newest]
Thread overview: 70+ messages / expand[flat|nested] mbox.gz Atom feed top
2009-08-19 23:07 [Qemu-devel] [PATCH v1 00/29] QMonitor Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 01/29] Introduce QObject Luiz Capitulino
2009-08-20 8:34 ` [Qemu-devel] " Paolo Bonzini
2009-08-20 14:17 ` Luiz Capitulino
2009-08-20 14:21 ` Paolo Bonzini
2009-08-20 15:12 ` Luiz Capitulino
2009-08-20 15:15 ` Paolo Bonzini
2009-08-20 8:51 ` Paolo Bonzini
2009-08-19 23:07 ` [Qemu-devel] [PATCH 02/29] Introduce QInt Luiz Capitulino
2009-08-20 7:51 ` [Qemu-devel] " Avi Kivity
2009-08-20 9:22 ` Paolo Bonzini
2009-08-20 13:24 ` Luiz Capitulino
2009-08-20 13:26 ` Avi Kivity
2009-08-20 14:51 ` Luiz Capitulino
2009-08-20 14:55 ` Avi Kivity
2009-08-20 15:14 ` Luiz Capitulino
2009-08-20 8:37 ` Paolo Bonzini
2009-08-20 13:44 ` Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 03/29] Introduce QString Luiz Capitulino
2009-08-20 8:41 ` [Qemu-devel] " Paolo Bonzini
2009-08-20 14:19 ` Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 04/29] Introduce QDict Luiz Capitulino
2009-08-20 7:57 ` [Qemu-devel] " Avi Kivity
2009-08-20 13:57 ` Luiz Capitulino
2009-08-20 14:07 ` Avi Kivity
2009-08-20 15:08 ` Luiz Capitulino
2009-08-20 15:26 ` Avi Kivity
2009-08-20 9:00 ` Paolo Bonzini
2009-08-20 14:14 ` Luiz Capitulino
2009-08-24 15:40 ` [Qemu-devel] " Markus Armbruster
2009-08-24 16:11 ` Luiz Capitulino [this message]
2009-08-19 23:07 ` [Qemu-devel] [PATCH 05/29] Add wrappers to functions used by the Monitor Luiz Capitulino
2009-08-24 16:47 ` Markus Armbruster
2009-08-19 23:07 ` [Qemu-devel] [PATCH 06/29] monitor: New format for handlers argument types Luiz Capitulino
2009-08-24 16:21 ` Markus Armbruster
2009-08-24 17:00 ` Luiz Capitulino
2009-09-04 14:06 ` Anthony Liguori
2009-09-04 14:48 ` Luiz Capitulino
2009-09-04 15:18 ` Markus Armbruster
2009-08-19 23:07 ` [Qemu-devel] [PATCH 07/29] monitor: Setup a QDict with arguments to handlers Luiz Capitulino
2009-08-24 16:25 ` Markus Armbruster
2009-08-24 17:07 ` Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 08/29] monitor: Export QDict header Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 09/29] monitor: Port handler_0 to use QDict Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 10/29] monitor: Port handler_1 " Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 11/29] monitor: Port handler_2 " Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 12/29] monitor: Port handler_3 " Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 13/29] monitor: Port handler_4 " Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 14/29] monitor: Port handler_5 " Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 15/29] monitor: Port handler_6 " Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 16/29] monitor: Port handler_7 " Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 17/29] monitor: Drop handler_8 and handler_9 Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 18/29] monitor: Port handler_10 to use QDict Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 19/29] monitor: Split monitor_handle_command() Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 20/29] monitor: Drop unused macros Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 21/29] monitor: Drop str_allocated[] Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 22/29] monitor: Drop args[] handling code Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 23/29] monitor: fail when 'i' type is greater than 32-bit Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 24/29] monitor: Update supported types documentation Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 25/29] Add check support Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 26/29] Introduce QInt unit-tests Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 27/29] Introduce QString unit-tests Luiz Capitulino
2009-08-19 23:07 ` [Qemu-devel] [PATCH 28/29] Introduce QDict test data file Luiz Capitulino
2009-08-19 23:08 ` [Qemu-devel] [PATCH 29/29] Introduce QDict unit-tests Luiz Capitulino
-- strict thread matches above, loose matches on Subject: below --
2015-09-12 0:22 [Qemu-devel] [PATCH 04/29] Introduce QDict Programmingkid
2015-09-12 4:10 ` Luiz Capitulino
2015-09-12 12:08 ` Programmingkid
2009-08-28 18:27 [Qemu-devel] [PATCH v3 00/29] QMonitor Luiz Capitulino
2009-08-28 18:27 ` [Qemu-devel] [PATCH 04/29] Introduce QDict Luiz Capitulino
2009-08-26 17:05 [Qemu-devel] [PATCH v2 00/29] QMonitor Luiz Capitulino
2009-08-26 17:05 ` [Qemu-devel] [PATCH 04/29] Introduce QDict Luiz Capitulino
2009-08-13 13:49 [Qemu-devel] [PATCH v0 00/29] QMonitor Luiz Capitulino
2009-08-13 13:50 ` [Qemu-devel] [PATCH 04/29] Introduce QDict Luiz Capitulino
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20090824131158.414fe394@doriath \
--to=lcapitulino@redhat.com \
--cc=aliguori@us.ibm.com \
--cc=armbru@redhat.com \
--cc=avi@redhat.com \
--cc=qemu-devel@nongnu.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.