[HB] Add a "blob" manager
diff --git a/src/hb-blob.c b/src/hb-blob.c
new file mode 100644
index 0000000..bcdf134
--- /dev/null
+++ b/src/hb-blob.c
@@ -0,0 +1,181 @@
+/*
+ * Copyright (C) 2009 Red Hat, Inc.
+ *
+ * This is part of HarfBuzz, an OpenType Layout engine library.
+ *
+ * Permission is hereby granted, without written agreement and without
+ * license or royalty fees, to use, copy, modify, and distribute this
+ * software and its documentation for any purpose, provided that the
+ * above copyright notice and the following two paragraphs appear in
+ * all copies of this software.
+ *
+ * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
+ * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
+ * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
+ * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
+ * DAMAGE.
+ *
+ * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
+ * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
+ * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
+ * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
+ *
+ * Red Hat Author(s): Behdad Esfahbod
+ */
+
+#include "hb-private.h"
+
+#include "hb-blob.h"
+#include "hb-refcount-private.h"
+
+struct _hb_blob_t {
+ const char *data;
+ unsigned int len;
+ hb_memory_mode_t mode;
+
+ hb_reference_count_t ref_count;
+ hb_destroy_func_t destroy;
+ void *user_data;
+};
+static hb_blob_t _hb_blob_nil = {
+ NULL,
+ 0,
+ HB_MEMORY_MODE_READONLY,
+
+ HB_REFERENCE_COUNT_INVALID,
+ NULL,
+ NULL
+};
+
+static void
+_hb_blob_destroy_user_data (hb_blob_t *blob)
+{
+ if (blob->destroy) {
+ blob->destroy (blob->user_data);
+ blob->destroy = NULL;
+ blob->user_data = NULL;
+ }
+}
+
+hb_blob_t *
+hb_blob_create (const char *data,
+ unsigned int len,
+ hb_memory_mode_t mode,
+ hb_destroy_func_t destroy,
+ void *user_data)
+{
+ hb_blob_t *blob;
+
+ blob = malloc (sizeof (hb_blob_t));
+ if (!blob) {
+ if (destroy)
+ destroy (user_data);
+ return &_hb_blob_nil;
+ }
+
+ blob->data = data;
+ blob->len = len;
+ blob->mode = mode;
+
+ HB_REFERENCE_COUNT_INIT (blob->ref_count, 1);
+ blob->destroy = destroy;
+ blob->user_data = user_data;
+
+ if (blob->mode == HB_MEMORY_MODE_DUPLICATE) {
+ blob->mode = HB_MEMORY_MODE_READONLY;
+ hb_blob_make_writeable (blob);
+ }
+
+ return blob;
+}
+
+hb_blob_t *
+hb_blob_reference (hb_blob_t *blob)
+{
+ if (blob == NULL || HB_REFERENCE_COUNT_IS_INVALID (blob->ref_count))
+ return blob;
+
+ assert (HB_REFERENCE_COUNT_HAS_REFERENCE (blob->ref_count));
+
+ _hb_reference_count_inc (blob->ref_count);
+
+ return blob;
+}
+
+void
+hb_blob_destroy (hb_blob_t *blob)
+{
+ if (blob == NULL || HB_REFERENCE_COUNT_IS_INVALID (blob->ref_count))
+ return;
+
+ assert (HB_REFERENCE_COUNT_HAS_REFERENCE (blob->ref_count));
+
+ if (!_hb_reference_count_dec_and_test (blob->ref_count))
+ return;
+
+ _hb_blob_destroy_user_data (blob);
+
+ free (blob);
+}
+
+const char *
+hb_blob_get_data (hb_blob_t *blob,
+ unsigned int *len)
+{
+ if (len)
+ *len = blob->len;
+
+ return blob->data;
+}
+
+hb_bool_t
+hb_blob_is_writeable (hb_blob_t *blob)
+{
+ return blob->mode == HB_MEMORY_MODE_WRITEABLE;
+}
+
+hb_bool_t
+hb_blob_try_writeable_inplace (hb_blob_t *blob)
+{
+ if (blob->mode == HB_MEMORY_MODE_READONLY_MAY_MAKE_WRITEABLE) {
+ /* XXX
+ * mprotect
+ blob->mode == HB_MEMORY_MODE_WRITEABLE;
+ */
+ }
+
+ return blob->mode == HB_MEMORY_MODE_WRITEABLE;
+}
+
+/* DANGER: May rebase or nullify */
+void
+hb_blob_make_writeable (hb_blob_t *blob)
+{
+ if (blob->mode == HB_MEMORY_MODE_READONLY_NEVER_DUPLICATE)
+ {
+ _hb_blob_destroy_user_data (blob);
+ blob->data = NULL;
+ blob->len = 0;
+ }
+ else if (blob->mode == HB_MEMORY_MODE_READONLY)
+ {
+ char *new_data;
+
+ new_data = malloc (blob->len);
+ if (new_data)
+ memcpy (new_data, blob->data, blob->len);
+
+ _hb_blob_destroy_user_data (blob);
+
+ if (!new_data) {
+ blob->data = NULL;
+ blob->len = 0;
+ } else
+ blob->data = new_data;
+
+ blob->mode = HB_MEMORY_MODE_WRITEABLE;
+ }
+ else
+ hb_blob_try_writeable_inplace (blob);
+}