summaryrefslogtreecommitdiff
path: root/pango/opentype
diff options
context:
space:
mode:
authorBehdad Esfahbod <behdad@behdad.org>2009-08-03 17:53:29 -0400
committerBehdad Esfahbod <behdad@behdad.org>2009-08-03 17:53:29 -0400
commitab5a714897efa61da57d8a7fc060b57a6ff4991c (patch)
tree5f6057861583f7de7ba7bd3ca2fb4791160f799a /pango/opentype
parent48cc6e95b7b16c56ebad8636204b0f9599400ac5 (diff)
downloadpango-ab5a714897efa61da57d8a7fc060b57a6ff4991c.tar.gz
[HB] Add sub-blobs
Diffstat (limited to 'pango/opentype')
-rw-r--r--pango/opentype/hb-blob.c162
-rw-r--r--pango/opentype/hb-blob.h12
2 files changed, 147 insertions, 27 deletions
diff --git a/pango/opentype/hb-blob.c b/pango/opentype/hb-blob.c
index cf9dd539..74ddb929 100644
--- a/pango/opentype/hb-blob.c
+++ b/pango/opentype/hb-blob.c
@@ -28,11 +28,17 @@
#include "hb-blob.h"
+#include <unistd.h>
+#include <sys/mman.h>
+
struct _hb_blob_t {
hb_reference_count_t ref_count;
+ hb_blob_t *parent;
+
const char *data;
- unsigned int len;
+ unsigned int offset;
+ unsigned int length;
hb_memory_mode_t mode;
hb_destroy_func_t destroy;
@@ -41,8 +47,11 @@ struct _hb_blob_t {
static hb_blob_t _hb_blob_nil = {
HB_REFERENCE_COUNT_INVALID, /* ref_count */
+ NULL, /* parent */
+
NULL, /* data */
- 0, /* len */
+ 0, /* offset */
+ 0, /* length */
HB_MEMORY_MODE_READONLY, /* mode */
NULL, /* destroy */
@@ -53,29 +62,71 @@ static void
_hb_blob_destroy_user_data (hb_blob_t *blob)
{
if (blob->destroy) {
+ if (blob->parent == blob->user_data)
+ blob->parent = NULL;
blob->destroy (blob->user_data);
blob->destroy = NULL;
blob->user_data = NULL;
}
}
+static void
+_hb_blob_nullify (hb_blob_t *blob)
+{
+ _hb_blob_destroy_user_data (blob);
+ blob->data = NULL;
+ blob->offset = 0;
+ blob->length = 0;
+}
+
+static void
+_hb_blob_sync_parent_mode (hb_blob_t *blob)
+{
+ if (blob->parent) {
+ if (blob->mode != HB_MEMORY_MODE_WRITEABLE && hb_blob_is_writeable (blob->parent))
+ blob->mode = HB_MEMORY_MODE_WRITEABLE;
+ }
+}
+
+static void
+_hb_blob_sync_parent_data (hb_blob_t *blob)
+{
+ if (blob->parent) {
+ const char *pdata;
+ unsigned int plength;
+
+ pdata = hb_blob_get_data (blob->parent, &plength);
+
+ if (pdata != blob->data) {
+ if (blob->offset >= plength) {
+ /* nothing left */
+ _hb_blob_nullify (blob);
+ } else {
+ blob->data = pdata;
+ blob->length = MIN (blob->length, plength - blob->offset);
+ }
+ }
+ }
+}
+
hb_blob_t *
hb_blob_create (const char *data,
- unsigned int len,
+ unsigned int length,
hb_memory_mode_t mode,
hb_destroy_func_t destroy,
void *user_data)
{
hb_blob_t *blob;
- if (!HB_OBJECT_DO_CREATE (blob)) {
+ if (!length || !HB_OBJECT_DO_CREATE (blob)) {
if (destroy)
destroy (user_data);
return &_hb_blob_nil;
}
blob->data = data;
- blob->len = len;
+ blob->offset = 0;
+ blob->length = length;
blob->mode = mode;
blob->destroy = destroy;
@@ -90,6 +141,37 @@ hb_blob_create (const char *data,
}
hb_blob_t *
+hb_blob_create_sub_blob (hb_blob_t *parent,
+ unsigned int offset,
+ unsigned int length)
+{
+ hb_blob_t *blob;
+
+ if (!length || !HB_OBJECT_DO_CREATE (blob))
+ return &_hb_blob_nil;
+
+ blob->parent = parent; /* we keep the ref in user_data */
+
+ blob->data = parent->data + 1; /* make sure they're not equal */
+ blob->offset = offset;
+ blob->length = length;
+ blob->mode = parent->mode;
+
+ blob->destroy = (hb_destroy_func_t) hb_blob_destroy;
+ blob->user_data = hb_blob_reference (parent);
+
+ _hb_blob_sync_parent_data (blob);
+
+ return blob;
+}
+
+hb_blob_t *
+hb_blob_create_empty (void)
+{
+ return &_hb_blob_nil;
+}
+
+hb_blob_t *
hb_blob_reference (hb_blob_t *blob)
{
HB_OBJECT_DO_REFERENCE (blob);
@@ -113,28 +195,50 @@ hb_blob_destroy (hb_blob_t *blob)
const char *
hb_blob_get_data (hb_blob_t *blob,
- unsigned int *len)
+ unsigned int *length)
{
- if (len)
- *len = blob->len;
+ _hb_blob_sync_parent_data (blob);
+
+ if (length)
+ *length = blob->length;
- return blob->data;
+ return blob->data + blob->offset;
}
hb_bool_t
hb_blob_is_writeable (hb_blob_t *blob)
{
+ _hb_blob_sync_parent_mode (blob);
+
return blob->mode == HB_MEMORY_MODE_WRITEABLE;
}
hb_bool_t
hb_blob_try_writeable_inplace (hb_blob_t *blob)
{
+ if (HB_OBJECT_IS_INERT (blob))
+ return FALSE;
+
+ _hb_blob_sync_parent_mode (blob);
+
if (blob->mode == HB_MEMORY_MODE_READONLY_MAY_MAKE_WRITEABLE) {
- /* XXX
- * mprotect
- blob->mode == HB_MEMORY_MODE_WRITEABLE;
- */
+ _hb_blob_sync_parent_data (blob);
+
+ if (blob->length) {
+ int pagesize;
+ unsigned int length;
+ const char *addr;
+
+ pagesize = sysconf(_SC_PAGE_SIZE);
+ if (-1 == pagesize)
+ return FALSE;
+
+ addr = (const char *) (((size_t) blob->data + blob->offset) & pagesize);
+ length = (const char *) (((size_t) blob->data + blob->offset + blob->length + pagesize-1) & pagesize) - addr;
+ if (-1 == mprotect ((void *) addr, length, PROT_READ | PROT_WRITE))
+ return FALSE;
+ }
+ blob->mode = HB_MEMORY_MODE_WRITEABLE;
}
return blob->mode == HB_MEMORY_MODE_WRITEABLE;
@@ -144,27 +248,35 @@ hb_blob_try_writeable_inplace (hb_blob_t *blob)
void
hb_blob_make_writeable (hb_blob_t *blob)
{
+ if (HB_OBJECT_IS_INERT (blob))
+ return;
+
+ _hb_blob_sync_parent_mode (blob);
+
if (blob->mode == HB_MEMORY_MODE_READONLY_NEVER_DUPLICATE)
{
- _hb_blob_destroy_user_data (blob);
- blob->data = NULL;
- blob->len = 0;
+ _hb_blob_nullify (blob);
}
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_sync_parent_data (blob);
+
+ if (blob->length) {
+ new_data = malloc (blob->length);
+ if (new_data)
+ memcpy (new_data, blob->data + blob->offset, blob->length);
- _hb_blob_destroy_user_data (blob);
+ _hb_blob_destroy_user_data (blob);
- if (!new_data) {
- blob->data = NULL;
- blob->len = 0;
- } else
- blob->data = new_data;
+ if (!new_data) {
+ _hb_blob_nullify (blob);
+ } else {
+ blob->data = new_data;
+ blob->offset = 0;
+ }
+ }
blob->mode = HB_MEMORY_MODE_WRITEABLE;
}
diff --git a/pango/opentype/hb-blob.h b/pango/opentype/hb-blob.h
index 804cacd6..4ea5620f 100644
--- a/pango/opentype/hb-blob.h
+++ b/pango/opentype/hb-blob.h
@@ -43,12 +43,20 @@ typedef struct _hb_blob_t hb_blob_t;
hb_blob_t *
hb_blob_create (const char *data,
- unsigned int len,
+ unsigned int length,
hb_memory_mode_t mode,
hb_destroy_func_t destroy,
void *user_data);
hb_blob_t *
+hb_blob_create_sub_blob (hb_blob_t *parent,
+ unsigned int offset,
+ unsigned int length);
+
+hb_blob_t *
+hb_blob_create_empty (void);
+
+hb_blob_t *
hb_blob_reference (hb_blob_t *blob);
unsigned int
@@ -59,7 +67,7 @@ hb_blob_destroy (hb_blob_t *blob);
const char *
hb_blob_get_data (hb_blob_t *blob,
- unsigned int *len);
+ unsigned int *length);
hb_bool_t
hb_blob_is_writeable (hb_blob_t *blob);