All of lore.kernel.org
 help / color / mirror / Atom feed
From: Markus Armbruster <armbru@redhat.com>
To: Eric Blake <eblake@redhat.com>
Cc: qemu-devel@nongnu.org, Kevin Wolf <kwolf@redhat.com>,
	"open list:Block layer core" <qemu-block@nongnu.org>,
	Chrysostomos Nanakos <chris@include.gr>,
	Jeff Cody <jcody@redhat.com>,
	Luiz Capitulino <lcapitulino@redhat.com>,
	Max Reitz <mreitz@redhat.com>
Subject: Re: [Qemu-devel] [PATCH v4 21/28] qstring: Add qstring_wrap_str()
Date: Thu, 02 Jun 2016 17:21:39 +0200	[thread overview]
Message-ID: <87y46nd3b0.fsf@dusky.pond.sub.org> (raw)
In-Reply-To: <1463632874-28559-22-git-send-email-eblake@redhat.com> (Eric Blake's message of "Wed, 18 May 2016 22:41:07 -0600")

Eric Blake <eblake@redhat.com> writes:

> Several spots in the code malloc a string, then wrap it in a
> QString, which has to duplicate the input.  Adding a new
> constructor with transfer semantics makes this pattern more
> efficient, comparable to the just-added transfer semantics to
> go from QString back to raw string.  Use the new
> qstring_wrap_str() where it makes sense.
>
> The new test still passes under valgrind.
>
> Signed-off-by: Eric Blake <eblake@redhat.com>
>
> ---
> v4: new patch
> ---
>  include/qapi/qmp/qstring.h |  1 +
>  block.c                    |  3 ++-
>  block/archipelago.c        |  6 ++----
>  blockdev.c                 |  3 +--
>  qobject/qstring.c          | 25 +++++++++++++++++++++++++
>  tests/check-qstring.c      | 24 ++++++++++++++++++++++++
>  6 files changed, 55 insertions(+), 7 deletions(-)
>
> diff --git a/include/qapi/qmp/qstring.h b/include/qapi/qmp/qstring.h
> index 2d55c87..97cf776 100644
> --- a/include/qapi/qmp/qstring.h
> +++ b/include/qapi/qmp/qstring.h
> @@ -25,6 +25,7 @@ typedef struct QString {
>  QString *qstring_new(void);
>  QString *qstring_from_str(const char *str);
>  QString *qstring_from_substr(const char *str, int start, int end);
> +QString *qstring_wrap_str(char *str);
>  size_t qstring_get_length(const QString *qstring);
>  const char *qstring_get_str(const QString *qstring);
>  char *qstring_consume_str(QString *qstring);
> diff --git a/block.c b/block.c
> index 551832f..32d06b5 100644
> --- a/block.c
> +++ b/block.c
> @@ -1448,7 +1448,8 @@ static int bdrv_append_temp_snapshot(BlockDriverState *bs, int flags,
>      qdict_put(snapshot_options, "file.driver",
>                qstring_from_str("file"));
>      qdict_put(snapshot_options, "file.filename",
> -              qstring_from_str(tmp_filename));
> +              qstring_wrap_str(tmp_filename));
> +    tmp_filename = NULL;
>      qdict_put(snapshot_options, "driver",
>                qstring_from_str("qcow2"));
>

The g_free(tmp_filename) stays, because there are error paths bypassing
this spot.  Good.

> diff --git a/block/archipelago.c b/block/archipelago.c
> index 104f2f9..bfdc14c 100644
> --- a/block/archipelago.c
> +++ b/block/archipelago.c
> @@ -426,13 +426,11 @@ static void archipelago_parse_filename(const char *filename, QDict *options,
>      parse_filename_opts(filename, errp, &volume, &segment_name, &mport, &vport);
>
>      if (volume) {
> -        qdict_put(options, ARCHIPELAGO_OPT_VOLUME, qstring_from_str(volume));
> -        g_free(volume);
> +        qdict_put(options, ARCHIPELAGO_OPT_VOLUME, qstring_wrap_str(volume));
>      }
>      if (segment_name) {
>          qdict_put(options, ARCHIPELAGO_OPT_SEGMENT,
> -                  qstring_from_str(segment_name));
> -        g_free(segment_name);
> +                  qstring_wrap_str(segment_name));
>      }
>      if (mport != NoPort) {
>          qdict_put(options, ARCHIPELAGO_OPT_MPORT, qint_from_int(mport));
> diff --git a/blockdev.c b/blockdev.c
> index adc5058..f293cb3 100644
> --- a/blockdev.c
> +++ b/blockdev.c
> @@ -1089,8 +1089,7 @@ DriveInfo *drive_new(QemuOpts *all_opts, BlockInterfaceType block_default_type)
>              new_id = g_strdup_printf("%s%s%i", if_name[type],
>                                       mediastr, unit_id);
>          }
> -        qdict_put(bs_opts, "id", qstring_from_str(new_id));
> -        g_free(new_id);
> +        qdict_put(bs_opts, "id", qstring_wrap_str(new_id));
>      }
>
>      /* Add virtio block device */
> diff --git a/qobject/qstring.c b/qobject/qstring.c
> index 7a438e9..2af1914 100644
> --- a/qobject/qstring.c
> +++ b/qobject/qstring.c
> @@ -66,6 +66,31 @@ QString *qstring_from_str(const char *str)
>      return qstring_from_substr(str, 0, strlen(str) - 1);
>  }
>
> +/**
> + * qstring_wrap_str(): Create a new QString around a malloc'd C string
> + *
> + * Returns a strong reference, and caller must not use @str any more.
> + * @str may be NULL, in which case the QString will be "".

I'm not fond of conflating null pointers and "" (see also the trouble
with visit_type_str()).  For what it's worth, qstring_from_str(NULL)
crashes.  Can we reject null?

> + */
> +QString *qstring_wrap_str(char *str)
> +{
> +    QString *qstring;
> +
> +    qstring = g_malloc(sizeof(*qstring));
> +    qobject_init(QOBJECT(qstring), QTYPE_QSTRING);
> +
> +    if (str) {
> +        qstring->string = str;
> +        qstring->length = strlen(str);
> +    } else {
> +        qstring->string = g_strdup("");
> +        qstring->length = 0;
> +    }
> +    qstring->capacity = qstring->length;
> +
> +    return qstring;
> +}
> +
>  static void capacity_increase(QString *qstring, size_t len)
>  {
>      if (qstring->capacity < (qstring->length + len)) {
> diff --git a/tests/check-qstring.c b/tests/check-qstring.c
> index e6f58e0..0e2e9bd 100644
> --- a/tests/check-qstring.c
> +++ b/tests/check-qstring.c
> @@ -35,6 +35,29 @@ static void qstring_from_str_test(void)
>      QDECREF(qstring);
>  }
>
> +static void qstring_wrap_str_test(void)
> +{
> +    QString *qstring;
> +    char *str = g_strdup("QEMU");
> +
> +    qstring = qstring_wrap_str(str);
> +    g_assert(qstring != NULL);
> +    g_assert(qstring->base.refcnt == 1);
> +    g_assert(strcmp("QEMU", qstring->string) == 0);
> +    g_assert(qobject_type(QOBJECT(qstring)) == QTYPE_QSTRING);
> +
> +    QDECREF(qstring);
> +
> +    str = NULL;
> +    qstring = qstring_wrap_str(str);
> +    g_assert(qstring != NULL);
> +    g_assert(qstring->base.refcnt == 1);
> +    g_assert(strcmp("", qstring->string) == 0);
> +    g_assert(qobject_type(QOBJECT(qstring)) == QTYPE_QSTRING);
> +
> +    QDECREF(qstring);
> +}
> +
>  static void qstring_destroy_test(void)
>  {
>      QString *qstring = qstring_from_str("destroy test");
> @@ -120,6 +143,7 @@ int main(int argc, char **argv)
>      g_test_init(&argc, &argv, NULL);
>
>      g_test_add_func("/public/from_str", qstring_from_str_test);
> +    g_test_add_func("/public/wrap_str", qstring_wrap_str_test);
>      g_test_add_func("/public/destroy", qstring_destroy_test);
>      g_test_add_func("/public/get_str", qstring_get_str_test);
>      g_test_add_func("/public/append_chr", qstring_append_chr_test);

  reply	other threads:[~2016-06-02 15:21 UTC|newest]

Thread overview: 73+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2016-05-19  4:40 [Qemu-devel] [PATCH v4 00/28] Add qapi-to-JSON and clone visitors Eric Blake
2016-05-19  4:40 ` [PATCH v4 01/28] qapi: Rename (one) qjson.h to qobject-json.h Eric Blake
2016-05-19  4:40   ` [Qemu-devel] " Eric Blake
2016-06-01 15:09   ` Markus Armbruster
2016-06-01 15:09     ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 02/28] qapi: Improve use of qmp/types.h Eric Blake
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 03/28] qemu-img: Don't leak errors when outputting JSON Eric Blake
2016-06-01 15:25   ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 04/28] qapi: Add parameter to visit_end_* Eric Blake
2016-06-01 15:36   ` Markus Armbruster
2016-06-07 23:20     ` Eric Blake
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 05/28] qapi: Add new visit_free() function Eric Blake
2016-06-01 16:03   ` Markus Armbruster
2016-06-03 11:46     ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 06/28] opts-visitor: Favor " Eric Blake
2016-06-01 16:06   ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 07/28] string-input-visitor: " Eric Blake
2016-06-01 16:13   ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 08/28] qmp-input-visitor: " Eric Blake
2016-06-01 16:19   ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 09/28] string-output-visitor: " Eric Blake
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 10/28] qmp-output-visitor: " Eric Blake
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 11/28] tests: Factor out common code in qapi output tests Eric Blake
2016-06-01 16:33   ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 12/28] qapi: Add new visit_complete() function Eric Blake
2016-06-01 17:02   ` Markus Armbruster
2016-05-19  4:40 ` [Qemu-devel] [PATCH v4 13/28] qapi: Add new clone visitor Eric Blake
2016-06-02 13:43   ` Markus Armbruster
2016-06-03 14:04     ` Markus Armbruster
2016-06-09  4:15     ` Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 14/28] sockets: Use new QAPI cloning Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 15/28] replay: " Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 16/28] qapi: Factor out JSON string escaping Eric Blake
2016-06-02 14:53   ` Markus Armbruster
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 17/28] qapi: Factor out JSON number formatting Eric Blake
2016-06-02 15:02   ` Markus Armbruster
2016-06-02 15:06     ` Eric Blake
2016-06-03  9:02       ` Markus Armbruster
2016-06-09 16:07         ` Eric Blake
2016-06-13  8:22           ` Markus Armbruster
2016-06-13 12:34             ` Eric Blake
2016-06-13 14:41               ` Markus Armbruster
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 18/28] qapi: Add qstring_append_printf() Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 19/28] qapi: Use qstring_append_chr() where appropriate Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 20/28] qstring: Add qstring_consume_str() Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 21/28] qstring: Add qstring_wrap_str() Eric Blake
2016-06-02 15:21   ` Markus Armbruster [this message]
2016-06-09 16:31     ` Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 22/28] qobject: Consolidate qobject_to_json() calls Eric Blake
2016-06-02 15:32   ` Markus Armbruster
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 23/28] tests: Test qobject_to_json() pretty formatting Eric Blake
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 24/28] qapi: Add JSON output visitor Eric Blake
2016-06-03  7:39   ` Markus Armbruster
2016-06-03 12:53     ` Eric Blake
2016-06-03 14:09       ` Markus Armbruster
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 25/28] qapi: Support pretty printing in " Eric Blake
2016-06-03  7:56   ` Markus Armbruster
2016-06-03 12:55     ` Eric Blake
2016-06-03 14:08       ` Markus Armbruster
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 26/28] qobject: Implement qobject_to_json() atop JSON visitor Eric Blake
2016-06-03  8:25   ` Markus Armbruster
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 27/28] qapi: Add 'any' support to JSON output Eric Blake
2016-06-03  8:29   ` Markus Armbruster
2016-05-19  4:41 ` [Qemu-devel] [PATCH v4 28/28] qemu-img: Use new JSON output formatter Eric Blake
2016-05-19 14:58 ` [Qemu-devel] [PATCH v4 00/28] Add qapi-to-JSON and clone visitors Eric Blake
2016-05-19 16:52 ` [Qemu-devel] [PATCH v4 29/28] qapi: Add strict mode to JSON output visitor Eric Blake
2016-05-19 20:18   ` Eric Blake
2016-06-03  8:36     ` Markus Armbruster
2016-06-03  9:21   ` Markus Armbruster
2016-05-19 17:05 ` [Qemu-devel] [PATCH v4 00/28] Add qapi-to-JSON and clone visitors Markus Armbruster
2016-06-03 12:09 ` Markus Armbruster
2016-06-09 16:16   ` Eric Blake
2016-06-13  8:26     ` Markus Armbruster

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=87y46nd3b0.fsf@dusky.pond.sub.org \
    --to=armbru@redhat.com \
    --cc=chris@include.gr \
    --cc=eblake@redhat.com \
    --cc=jcody@redhat.com \
    --cc=kwolf@redhat.com \
    --cc=lcapitulino@redhat.com \
    --cc=mreitz@redhat.com \
    --cc=qemu-block@nongnu.org \
    --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.