dba_cdb.c 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Sascha Schumann <sascha@schumann.cx> |
  16. | Marcus Boerger <helly@php.net> |
  17. +----------------------------------------------------------------------+
  18. */
  19. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #if DBA_CDB
  24. #include "php_cdb.h"
  25. #include <sys/types.h>
  26. #ifdef HAVE_UNISTD_H
  27. #include <unistd.h>
  28. #endif
  29. #include <fcntl.h>
  30. #if DBA_CDB_BUILTIN
  31. # include "libcdb/cdb.h"
  32. # include "libcdb/cdb_make.h"
  33. # include "libcdb/uint32.h"
  34. #else
  35. # ifdef CDB_INCLUDE_FILE
  36. # include CDB_INCLUDE_FILE
  37. # endif
  38. #endif
  39. #define CDB_INFO \
  40. dba_cdb *cdb = (dba_cdb *) info->dbf
  41. typedef struct {
  42. struct cdb c;
  43. #if DBA_CDB_BUILTIN
  44. struct cdb_make m;
  45. php_stream *file;
  46. int make;
  47. #else
  48. int file;
  49. #endif
  50. uint32 eod; /* size of constant database */
  51. uint32 pos; /* current position for traversing */
  52. } dba_cdb;
  53. DBA_OPEN_FUNC(cdb)
  54. {
  55. #if DBA_CDB_BUILTIN
  56. php_stream* file = 0;
  57. int make;
  58. #else
  59. int file = 0;
  60. #endif
  61. dba_cdb *cdb;
  62. dba_info *pinfo = (dba_info *) info;
  63. switch (info->mode) {
  64. case DBA_READER:
  65. #if DBA_CDB_BUILTIN
  66. make = 0;
  67. file = info->fp;
  68. #else
  69. file = VCWD_OPEN(info->path, O_RDONLY);
  70. if (file < 0) {
  71. *error = "Unable to open file";
  72. return FAILURE;
  73. }
  74. #endif
  75. break;
  76. #if DBA_CDB_BUILTIN
  77. case DBA_TRUNC:
  78. make = 1;
  79. file = info->fp;
  80. break;
  81. case DBA_CREAT:
  82. case DBA_WRITER:
  83. *error = "Update operations are not supported";
  84. return FAILURE; /* not supported */
  85. #endif
  86. default:
  87. *error = "Currently not supported";
  88. return FAILURE;
  89. }
  90. cdb = pemalloc(sizeof(dba_cdb), info->flags&DBA_PERSISTENT);
  91. memset(cdb, 0, sizeof(dba_cdb));
  92. #if DBA_CDB_BUILTIN
  93. if (make) {
  94. cdb_make_start(&cdb->m, file);
  95. } else {
  96. cdb_init(&cdb->c, file);
  97. }
  98. cdb->make = make;
  99. #else
  100. cdb_init(&cdb->c, file);
  101. #endif
  102. cdb->file = file;
  103. pinfo->dbf = cdb;
  104. return SUCCESS;
  105. }
  106. DBA_CLOSE_FUNC(cdb)
  107. {
  108. CDB_INFO;
  109. /* cdb_free does not close associated file */
  110. #if DBA_CDB_BUILTIN
  111. if (cdb->make) {
  112. cdb_make_finish(&cdb->m);
  113. } else {
  114. cdb_free(&cdb->c);
  115. }
  116. #else
  117. cdb_free(&cdb->c);
  118. close(cdb->file);
  119. #endif
  120. pefree(cdb, info->flags&DBA_PERSISTENT);
  121. }
  122. #if DBA_CDB_BUILTIN
  123. # define php_cdb_read(cdb, buf, len, pos) cdb_read(cdb, buf, len, pos)
  124. # define php_cdb_findnext(cdb, key, len) cdb_findnext(cdb, key, len)
  125. # define php_cdb_find(cdb, key, len) cdb_find(cdb, key, len)
  126. #else
  127. # define php_cdb_read(cdb, buf, len, pos) cdb_read(cdb, buf, len, pos)
  128. # define php_cdb_findnext(cdb, key, len) cdb_findnext(cdb, key, len)
  129. # define php_cdb_find(cdb, key, len) cdb_find(cdb, key, len)
  130. #endif
  131. DBA_FETCH_FUNC(cdb)
  132. {
  133. CDB_INFO;
  134. unsigned int len;
  135. char *new_entry = NULL;
  136. #if DBA_CDB_BUILTIN
  137. if (cdb->make)
  138. return NULL; /* database was opened writeonly */
  139. #endif
  140. if (php_cdb_find(&cdb->c, key, keylen) == 1) {
  141. while(skip--) {
  142. if (php_cdb_findnext(&cdb->c, key, keylen) != 1) {
  143. return NULL;
  144. }
  145. }
  146. len = cdb_datalen(&cdb->c);
  147. new_entry = safe_emalloc(len, 1, 1);
  148. if (php_cdb_read(&cdb->c, new_entry, len, cdb_datapos(&cdb->c)) == -1) {
  149. efree(new_entry);
  150. return NULL;
  151. }
  152. new_entry[len] = 0;
  153. if (newlen)
  154. *newlen = len;
  155. }
  156. return new_entry;
  157. }
  158. DBA_UPDATE_FUNC(cdb)
  159. {
  160. #if DBA_CDB_BUILTIN
  161. CDB_INFO;
  162. if (!cdb->make)
  163. return FAILURE; /* database was opened readonly */
  164. if (!mode)
  165. return FAILURE; /* cdb_make dosn't know replace */
  166. if (cdb_make_add(&cdb->m, key, keylen, val, vallen) != -1)
  167. return SUCCESS;
  168. #endif
  169. return FAILURE;
  170. }
  171. DBA_EXISTS_FUNC(cdb)
  172. {
  173. CDB_INFO;
  174. #if DBA_CDB_BUILTIN
  175. if (cdb->make)
  176. return FAILURE; /* database was opened writeonly */
  177. #endif
  178. if (php_cdb_find(&cdb->c, key, keylen) == 1)
  179. return SUCCESS;
  180. return FAILURE;
  181. }
  182. DBA_DELETE_FUNC(cdb)
  183. {
  184. return FAILURE; /* cdb doesn't support delete */
  185. }
  186. /* {{{ cdb_file_read */
  187. #if DBA_CDB_BUILTIN
  188. # define cdb_file_read(fildes, buf, size) php_stream_read(fildes, buf, size)
  189. #else
  190. # define cdb_file_read(fildes, buf, size) read(fildes, buf, size)
  191. #endif
  192. /* }}} */
  193. #define CREAD(n) do { \
  194. if (cdb_file_read(cdb->file, buf, n) < n) return NULL; \
  195. } while (0)
  196. /* {{{ cdb_file_lseek
  197. php_stream_seek does not return actual position */
  198. #if DBA_CDB_BUILTIN
  199. zend_off_t cdb_file_lseek(php_stream *fp, zend_off_t offset, int whence) {
  200. php_stream_seek(fp, offset, whence);
  201. return php_stream_tell(fp);
  202. }
  203. #else
  204. zend_off_t cdb_file_lseek(int fd, zend_off_t offset, int whence) {
  205. return lseek(fd, offset, whence);
  206. }
  207. #endif
  208. /* }}} */
  209. #define CSEEK(n) do { \
  210. if (n >= cdb->eod) return NULL; \
  211. if (cdb_file_lseek(cdb->file, (zend_off_t)n, SEEK_SET) != (zend_off_t) n) return NULL; \
  212. } while (0)
  213. DBA_FIRSTKEY_FUNC(cdb)
  214. {
  215. CDB_INFO;
  216. uint32 klen, dlen;
  217. char buf[8];
  218. char *key;
  219. #if DBA_CDB_BUILTIN
  220. if (cdb->make)
  221. return NULL; /* database was opened writeonly */
  222. #endif
  223. cdb->eod = -1;
  224. CSEEK(0);
  225. CREAD(4);
  226. /* Total length of file in bytes */
  227. uint32_unpack(buf, &cdb->eod);
  228. CSEEK(2048);
  229. CREAD(8);
  230. /* The first four bytes contain the length of the key */
  231. uint32_unpack(buf, &klen);
  232. uint32_unpack(buf + 4, &dlen);
  233. key = safe_emalloc(klen, 1, 1);
  234. if (cdb_file_read(cdb->file, key, klen) < klen) {
  235. efree(key);
  236. key = NULL;
  237. } else {
  238. key[klen] = '\0';
  239. if (newlen) *newlen = klen;
  240. }
  241. /* header + klenlen + dlenlen + klen + dlen */
  242. cdb->pos = 2048 + 4 + 4 + klen + dlen;
  243. return key;
  244. }
  245. DBA_NEXTKEY_FUNC(cdb)
  246. {
  247. CDB_INFO;
  248. uint32 klen, dlen;
  249. char buf[8];
  250. char *key;
  251. #if DBA_CDB_BUILTIN
  252. if (cdb->make)
  253. return NULL; /* database was opened writeonly */
  254. #endif
  255. CSEEK(cdb->pos);
  256. CREAD(8);
  257. uint32_unpack(buf, &klen);
  258. uint32_unpack(buf + 4, &dlen);
  259. key = safe_emalloc(klen, 1, 1);
  260. if (cdb_file_read(cdb->file, key, klen) < klen) {
  261. efree(key);
  262. key = NULL;
  263. } else {
  264. key[klen] = '\0';
  265. if (newlen) *newlen = klen;
  266. }
  267. cdb->pos += 8 + klen + dlen;
  268. return key;
  269. }
  270. DBA_OPTIMIZE_FUNC(cdb)
  271. {
  272. return SUCCESS;
  273. }
  274. DBA_SYNC_FUNC(cdb)
  275. {
  276. /* this is read-only */
  277. return SUCCESS;
  278. }
  279. DBA_INFO_FUNC(cdb)
  280. {
  281. #if DBA_CDB_BUILTIN
  282. if (!strcmp(hnd->name, "cdb")) {
  283. return estrdup(cdb_version());
  284. } else {
  285. return estrdup(cdb_make_version());
  286. }
  287. #else
  288. return estrdup("External");
  289. #endif
  290. }
  291. #endif
  292. /*
  293. * Local variables:
  294. * tab-width: 4
  295. * c-basic-offset: 4
  296. * End:
  297. * vim600: sw=4 ts=4 fdm=marker
  298. * vim<600: sw=4 ts=4
  299. */