update.
[chise/libchise.git] / chise.c
diff --git a/chise.c b/chise.c
index 3397991..c37a61a 100644 (file)
--- a/chise.c
+++ b/chise.c
@@ -22,186 +22,119 @@ strnlen (register const char *s, register int maxlen)
 
 #define xzero(lvalue) ((void) memset (&(lvalue), '\0', sizeof (lvalue)))
 
-
-int
-chise_open_data_source (CHISE_DS *ds, CHISE_DS_Type type, char *location)
-{
-  ds->type = type;
-  ds->location = location;
-  return 0;
-}
-
-int
-chise_close_data_source (CHISE_DS *ds)
-{
-  ds->type = CHISE_DS_NONE;
-  ds->location = NULL;
-  return 0;
-}
-
-
-int
-chise_open_decoding_table (CHISE_Decoding_Table **db,
-                          CHISE_DS *ds, const char *ccs,
-                          DBTYPE real_subtype,
-                          u_int32_t accessmask, int modemask)
-{
-  return
-    chise_open_attribute_table (db, ds->location,
-                               ccs, "system-char-id",
-                               real_subtype, accessmask, modemask);
-}
-
-int
-chise_close_decoding_table (CHISE_Decoding_Table *db)
-{
-  if (db)
-    return chise_close_attribute_table (db);
-  return -1;
-}
-
 CHISE_Char_ID
-chise_dt_get_char (CHISE_Decoding_Table *db, int code_point)
+chise_char_id_parse_c_string (unsigned char *str, int len)
 {
-  CHISE_Value valdatum;
-  int status = 0;
-  char key_buf[16];
+  int i = 0;
 
-  sprintf(key_buf, "%d", code_point);
-  status = chise_get_attribute_table (db, key_buf, &valdatum);
-  if (!status)
+  if ( (len >= 2) && (str[i++] == '?') )
     {
-      unsigned char *str
-       = (unsigned char *)chise_value_data (&valdatum);
-      int len = strnlen (str, chise_value_size (&valdatum));
-      int i = 0;
+      unsigned char c = str[i++];
+      int counter;
+      CHISE_Char_ID cid;
 
-      if ( (len >= 2) && (str[i++] == '?') )
+      if (c == '\\')
        {
-         unsigned char c = str[i++];
-         int counter;
-         CHISE_Char_ID cid;
-
-         if (c == '\\')
+         if (len < 3)
+           return -1;
+         c = str[i++];
+         if (c == '^')
            {
-             if (len < 3)
+             if (len < 4)
                return -1;
              c = str[i++];
-             if (c == '^')
-               {
-                 if (len < 4)
-                   return -1;
-                 c = str[i++];
-                 if (c == '?')
-                   return 0x7F;
-                 else
-                   return c & (0x80 | 0x1F);
-               }
-           }
-         if ( c < 0xC0 )
-           {
-             cid = c;
-             counter = 0;
-           }
-         else if ( c < 0xE0 )
-           {
-             cid = c & 0x1f;
-             counter = 1;
-           }
-         else if ( c < 0xF0 )
-           {
-             cid = c & 0x0f;
-             counter = 2;
-           }
-         else if ( c < 0xF8 )
-           {
-             cid = c & 0x07;
-             counter = 3;
-           }
-         else if ( c < 0xFC )
-           {
-             cid = c & 0x03;
-             counter = 4;
-           }
-         else
-           {
-             cid = c & 0x01;
-             counter = 5;
+             if (c == '?')
+               return 0x7F;
+             else
+               return c & (0x80 | 0x1F);
            }
+       }
+      if ( c < 0xC0 )
+       {
+         cid = c;
+         counter = 0;
+       }
+      else if ( c < 0xE0 )
+       {
+         cid = c & 0x1f;
+         counter = 1;
+       }
+      else if ( c < 0xF0 )
+       {
+         cid = c & 0x0f;
+         counter = 2;
+       }
+      else if ( c < 0xF8 )
+       {
+         cid = c & 0x07;
+         counter = 3;
+       }
+      else if ( c < 0xFC )
+       {
+         cid = c & 0x03;
+         counter = 4;
+       }
+      else
+       {
+         cid = c & 0x01;
+         counter = 5;
+       }
 
-         if (counter + 2 <= len)
-           {
-             int j;
+      if (counter + 2 <= len)
+       {
+         int j;
 
-             for (j = 0; j < counter; j++)
-               cid = (cid << 6) | (str[j + i] & 0x3F);
-             return cid;
-           }
+         for (j = 0; j < counter; j++)
+           cid = (cid << 6) | (str[j + i] & 0x3F);
+         return cid;
        }
     }
   return -1;
 }
 
-
-
 int
-chise_open_feature_table (CHISE_Feature_Table **db,
-                         CHISE_DS *ds, const char *feature,
-                         DBTYPE real_subtype,
-                         u_int32_t accessmask, int modemask)
+chise_format_char_id (CHISE_Char_ID cid, unsigned char *dest, int len)
 {
-  return
-    chise_open_attribute_table (db, ds->location,
-                               "system-char-id", feature,
-                               real_subtype, accessmask, modemask);
-}
-
-int
-chise_close_feature_table (CHISE_Feature_Table *db)
-{
-  if (db)
-    return chise_close_attribute_table (db);
-  return -1;
-}
-
-int chise_ft_get_value (CHISE_Feature_Table *db,
-                       CHISE_Char_ID cid, CHISE_Value *valdatum)
-{
-  unsigned char key_buf[8];
+  int i = 0;
 
-  key_buf[0] = '?';
+  dest[i++] = '?';
   if (cid == '\t')
     {
-      key_buf[1] = '\\';
-      key_buf[2] = 't';
-      key_buf[3] = '\0';
+      dest[i++] = '\\';
+      dest[i++] = 't';
+      dest[i] = '\0';
+      return i;
     }
   else if (cid == '\n')
     {
-      key_buf[1] = '\\';
-      key_buf[2] = 'n';
-      key_buf[3] = '\0';
+      dest[i++] = '\\';
+      dest[i++] = 'n';
+      dest[i] = '\0';
+      return i;
     }
   else if (cid == '\r')
     {
-      key_buf[1] = '\\';
-      key_buf[2] = 'r';
-      key_buf[3] = '\0';
+      dest[i++] = '\\';
+      dest[i++] = 'r';
+      dest[i] = '\0';
+      return i;
     }
   else if (cid == 0x1C)
     {
-      key_buf[1] = '\\';
-      key_buf[2] = '^';
-      key_buf[3] = '\\';
-      key_buf[4] = '\\';
-      key_buf[5] = '\0';
+      dest[i++] = '\\';
+      dest[i++] = '^';
+      dest[i++] = '\\';
+      dest[i++] = '\\';
+      dest[i] = '\0';
+      return i;
     }
   else if (cid <= 0x1F)
     {
-      key_buf[1] = '\\';
-      key_buf[2] = '^';
-      key_buf[3] = '@' + cid;
-      key_buf[4] = '\0';
+      dest[i++] = '\\';
+      dest[i++] = '^';
+      dest[i++] = '@' + cid;
+      dest[i] = '\0';
+      return i;
     }
   else if ( (cid == ' ') || (cid == '"') ||
            (cid == '#') || (cid == '\'') ||
@@ -211,93 +144,252 @@ int chise_ft_get_value (CHISE_Feature_Table *db,
            (cid == '[') || (cid == '\\') ||
            (cid == ']') || (cid == '`') )
     {
-      key_buf[1] = '\\';
-      key_buf[2] = cid;
-      key_buf[3] = '\0';
+      dest[i++] = '\\';
+      dest[i++] = cid;
+      dest[i] = '\0';
+      return i;
     }
   else if (cid <= 0x7E)
     {
-      key_buf[1] = cid;
-      key_buf[2] = '\0';
+      dest[i++] = cid;
+      dest[i] = '\0';
+      return i;
     }
   else if (cid == 0x7F)
     {
-      key_buf[1] = '\\';
-      key_buf[2] = '^';
-      key_buf[3] = '?';
-      key_buf[4] = '\0';
+      dest[i++] = '\\';
+      dest[i++] = '^';
+      dest[i++] = '?';
+      dest[i] = '\0';
+      return i;
     }
   else if (cid <= 0x9F)
     {
-      key_buf[1] = '\\';
-      key_buf[2] = '^';
-      key_buf[3] = ((cid + '@') >> 6) | 0xC0;
-      key_buf[4] = ((cid + '@') & 0x3F) | 0x80;
-      key_buf[5] = '\0';
-
+      dest[i++] = '\\';
+      dest[i++] = '^';
+      dest[i++] = ((cid + '@') >> 6) | 0xC0;
+      dest[i++] = ((cid + '@') & 0x3F) | 0x80;
+      dest[i] = '\0';
+      return i;
     }
   else if (cid <= 0x7FF)
     {
-      key_buf[1] = (cid >> 6) | 0xC0;
-      key_buf[2] = (cid & 0x3F) | 0x80;
-      key_buf[3] = '\0';
+      dest[i++] = (cid >> 6) | 0xC0;
+      dest[i++] = (cid & 0x3F) | 0x80;
+      dest[i] = '\0';
+      return i;
     }
   else if (cid <= 0xFFFF)
     {
-      key_buf[1] = (cid >> 12) | 0xE0;
-      key_buf[2]= ((cid >>  6) & 0x3F) | 0x80;
-      key_buf[3]=  (cid        & 0x3F) | 0x80;
-      key_buf[4] = '\0';
+      dest[i++] = (cid >> 12) | 0xE0;
+      dest[i++]= ((cid >>  6) & 0x3F) | 0x80;
+      dest[i++]=  (cid        & 0x3F) | 0x80;
+      dest[i] = '\0';
+      return i;
     }
   else if (cid <= 0x1FFFFF)
     {
-      key_buf[1]=  (cid >> 18) | 0xF0;
-      key_buf[2]= ((cid >> 12) & 0x3F) | 0x80;
-      key_buf[3]= ((cid >>  6) & 0x3F) | 0x80;
-      key_buf[4]=  (cid        & 0x3F) | 0x80;
-      key_buf[5] = '\0';
+      dest[i++]=  (cid >> 18) | 0xF0;
+      dest[i++]= ((cid >> 12) & 0x3F) | 0x80;
+      dest[i++]= ((cid >>  6) & 0x3F) | 0x80;
+      dest[i++]=  (cid        & 0x3F) | 0x80;
+      dest[i] = '\0';
+      return i;
     }
   else if (cid <= 0x3FFFFFF)
     {
-      key_buf[1]=  (cid >> 24) | 0xF8;
-      key_buf[2]= ((cid >> 18) & 0x3F) | 0x80;
-      key_buf[3]= ((cid >> 12) & 0x3F) | 0x80;
-      key_buf[4]= ((cid >>  6) & 0x3F) | 0x80;
-      key_buf[5]=  (cid        & 0x3F) | 0x80;
-      key_buf[6] = '\0';
+      dest[i++]=  (cid >> 24) | 0xF8;
+      dest[i++]= ((cid >> 18) & 0x3F) | 0x80;
+      dest[i++]= ((cid >> 12) & 0x3F) | 0x80;
+      dest[i++]= ((cid >>  6) & 0x3F) | 0x80;
+      dest[i++]=  (cid        & 0x3F) | 0x80;
+      dest[i] = '\0';
+      return i;
     }
   else
     {
-      key_buf[1]=  (cid >> 30) | 0xFC;
-      key_buf[2]= ((cid >> 24) & 0x3F) | 0x80;
-      key_buf[3]= ((cid >> 18) & 0x3F) | 0x80;
-      key_buf[4]= ((cid >> 12) & 0x3F) | 0x80;
-      key_buf[5]= ((cid >>  6) & 0x3F) | 0x80;
-      key_buf[6]=  (cid        & 0x3F) | 0x80;
-      key_buf[7] = '\0';
+      dest[i++]=  (cid >> 30) | 0xFC;
+      dest[i++]= ((cid >> 24) & 0x3F) | 0x80;
+      dest[i++]= ((cid >> 18) & 0x3F) | 0x80;
+      dest[i++]= ((cid >> 12) & 0x3F) | 0x80;
+      dest[i++]= ((cid >>  6) & 0x3F) | 0x80;
+      dest[i++]=  (cid        & 0x3F) | 0x80;
+      dest[i] = '\0';
+      return i;
+    }
+}
+
+CHISE_DS*
+chise_open_data_source (CHISE_DS_Type type, char *location)
+{
+  CHISE_DS *ds = (CHISE_DS*)malloc (sizeof (CHISE_DS));
+  size_t len = strlen (location);
+
+  if (ds == NULL)
+    return NULL;
+
+  ds->type = type;
+  ds->location = (unsigned char*)malloc (len + 1);
+  if (ds->location == NULL)
+    {
+      free (ds);
+      return NULL;
     }
+  strcpy (ds->location, location);
+  return ds;
+}
+
+int
+chise_ds_close (CHISE_DS *ds)
+{
+  if (ds->location != NULL)
+    free (ds->location);
+  free (ds);
+  return 0;
+}
+
+
+CHISE_Decoding_Table*
+chise_ds_open_decoding_table (CHISE_DS *ds, const char *ccs,
+                             DBTYPE real_subtype,
+                             u_int32_t accessmask, int modemask)
+{
+  return
+    chise_open_attribute_table (ds,
+                               ccs, "system-char-id",
+                               real_subtype, accessmask, modemask);
+}
+
+int
+chise_dt_close (CHISE_Decoding_Table *table)
+{
+  if (table)
+    return chise_close_attribute_table (table);
+  return -1;
+}
+
+CHISE_Char_ID
+chise_dt_get_char (CHISE_Decoding_Table *table, int code_point)
+{
+  CHISE_Value valdatum;
+  int status = 0;
+  char key_buf[16];
+
+  sprintf(key_buf, "%d", code_point);
+  status = chise_get_attribute_table (table, key_buf, &valdatum);
+  if (!status)
+    {
+      unsigned char *str
+       = (unsigned char *)chise_value_data (&valdatum);
+      int len = strnlen (str, chise_value_size (&valdatum));
+
+      return chise_char_id_parse_c_string (str, len);
+    }
+  return -1;
+}
+
+int
+chise_dt_put_char (CHISE_Decoding_Table *table,
+                  int code_point, CHISE_Char_ID cid)
+{
+  CHISE_Value valdatum;
+  char key_buf[16], val_buf[8];
+
+  sprintf(key_buf, "%d", code_point);
+  chise_format_char_id (cid, val_buf, 8);
+  return chise_put_attribute_table (table, key_buf, val_buf);
+}
+
+
+
+CHISE_Feature_Table*
+chise_ds_open_feature_table (CHISE_DS *ds, const char *feature,
+                            DBTYPE real_subtype,
+                            u_int32_t accessmask, int modemask)
+{
   return
-    chise_get_attribute_table (db, key_buf, valdatum);
+    chise_open_attribute_table (ds,
+                               "system-char-id", feature,
+                               real_subtype, accessmask, modemask);
 }
 
+int
+chise_ft_close (CHISE_Feature_Table *table)
+{
+  if (table)
+    return chise_close_attribute_table (table);
+  return -1;
+}
 
 int
-chise_open_attribute_table (CHISE_Attribute_Table **db,
-                           const char *db_dir,
+chise_ft_get_value (CHISE_Feature_Table *table,
+                   CHISE_Char_ID cid, CHISE_Value *valdatum)
+{
+  unsigned char key_buf[8];
+
+  chise_format_char_id (cid, key_buf, 8);
+  return chise_get_attribute_table (table, key_buf, valdatum);
+}
+
+void
+chise_ft_iterate (CHISE_Feature_Table *table,
+                 int (*func) (CHISE_Feature_Table *table,
+                              CHISE_Char_ID cid, CHISE_Value *valdatum))
+{
+  DBT keydatum, valdatum;
+  DBC *dbcp;
+  int status;
+
+  xzero (keydatum);
+  xzero (valdatum);
+
+  status = table->db->cursor (table->db, NULL, &dbcp, 0);
+  for (status = dbcp->c_get (dbcp, &keydatum, &valdatum, DB_FIRST);
+       status == 0;
+       status = dbcp->c_get (dbcp, &keydatum, &valdatum, DB_NEXT))
+    {
+      unsigned char *key_str = (unsigned char *)keydatum.data;
+      int key_len = strnlen (key_str, keydatum.size);
+      CHISE_Char_ID key = chise_char_id_parse_c_string (key_str, key_len);
+      int ret;
+
+      if (ret = func (table, key, &valdatum))
+       break;
+    }
+  dbcp->c_close (dbcp);
+}
+
+CHISE_Attribute_Table*
+chise_open_attribute_table (CHISE_DS *ds,
                            const char *encoding, const char *feature,
                            DBTYPE real_subtype,
                            u_int32_t accessmask, int modemask)
 {
+  CHISE_Attribute_Table* table;
+  char *db_dir;
   DB* dbase;
   int status;
   int len, flen, i;
   int size;
   char *db_file_name, *sp;
 
+  if (ds == NULL)
+    return NULL;
+
+  table = (CHISE_Attribute_Table*)malloc (sizeof (CHISE_Attribute_Table));
+  if (table == NULL)
+    return NULL;
+
+  table->ds = ds;
+
   status = db_create (&dbase, NULL, 0);
   if (status)
-    return -1;
+    {
+      free (table);
+      return NULL;
+    }
 
+  db_dir = ds->location;
   len = strlen (db_dir);
   flen = strlen (feature);
   size = len + strlen (encoding) + flen * 3 + 4;
@@ -330,25 +422,27 @@ chise_open_attribute_table (CHISE_Attribute_Table **db,
   if (status)
     {
       dbase->close (dbase, 0);
-      return -1;
+      free (table);
+      return NULL;
     }
-  *db = dbase;
-  return 0;
+  table->db = dbase;
+  return table;
 }
 
 int
-chise_close_attribute_table (CHISE_Attribute_Table *db)
+chise_close_attribute_table (CHISE_Attribute_Table *table)
 {
-  if (db)
+  if (table->db)
     {
-      db->sync  (db, 0);
-      db->close (db, 0);
+      table->db->sync  (table->db, 0);
+      table->db->close (table->db, 0);
     }
+  free (table);
   return 0;
 }
 
 int
-chise_get_attribute_table (CHISE_Attribute_Table *db,
+chise_get_attribute_table (CHISE_Attribute_Table *table,
                           char *key, CHISE_Value *valdatum)
 {
   DBT keydatum;
@@ -361,6 +455,27 @@ chise_get_attribute_table (CHISE_Attribute_Table *db,
   keydatum.data = key;
   keydatum.size = strlen (key);
 
-  status = db->get (db, NULL, &keydatum, valdatum, 0);
+  status = table->db->get (table->db, NULL, &keydatum, valdatum, 0);
+  return status;
+}
+
+int
+chise_put_attribute_table (CHISE_Attribute_Table *table,
+                          char *key, char *value)
+{
+  DBT keydatum, valdatum;
+  int status = 0;
+
+  /* DB Version 2 requires DBT's to be zeroed before use. */
+  xzero (keydatum);
+  xzero (valdatum);
+
+  keydatum.data = key;
+  keydatum.size = strlen (key);
+
+  valdatum.data = value;
+  valdatum.size = strlen (value);
+
+  status = table->db->put (table->db, NULL, &keydatum, &valdatum, 0);
   return status;
 }