ibase_blobs.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  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: Ard Biesheuvel <a.k.biesheuvel@its.tudelft.nl> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #ifdef HAVE_CONFIG_H
  19. #include "config.h"
  20. #endif
  21. #include "php.h"
  22. #if HAVE_IBASE
  23. #include "php_interbase.h"
  24. #include "php_ibase_includes.h"
  25. #define BLOB_CLOSE 1
  26. #define BLOB_CANCEL 2
  27. #define PARSE_PARAMETERS \
  28. switch (ZEND_NUM_ARGS()) { \
  29. default: \
  30. WRONG_PARAM_COUNT; \
  31. case 1: \
  32. if (FAILURE == zend_parse_parameters(1, "s", &blob_id, &blob_id_len)) { \
  33. RETURN_FALSE; \
  34. } \
  35. break; \
  36. case 2: \
  37. if (FAILURE == zend_parse_parameters(2, "rs", &link, &blob_id, &blob_id_len)) { \
  38. RETURN_FALSE; \
  39. } \
  40. break; \
  41. } \
  42. static int le_blob;
  43. static void _php_ibase_free_blob(zend_resource *rsrc) /* {{{ */
  44. {
  45. ibase_blob *ib_blob = (ibase_blob *)rsrc->ptr;
  46. if (ib_blob->bl_handle != 0) { /* blob open*/
  47. if (isc_cancel_blob(IB_STATUS, &ib_blob->bl_handle)) {
  48. _php_ibase_module_error("You can lose data. Close any blob after reading from or "
  49. "writing to it. Use ibase_blob_close() before calling ibase_close()");
  50. }
  51. }
  52. efree(ib_blob);
  53. }
  54. /* }}} */
  55. void php_ibase_blobs_minit(INIT_FUNC_ARGS) /* {{{ */
  56. {
  57. le_blob = zend_register_list_destructors_ex(_php_ibase_free_blob, NULL,
  58. "interbase blob", module_number);
  59. }
  60. /* }}} */
  61. int _php_ibase_string_to_quad(char const *id, ISC_QUAD *qd) /* {{{ */
  62. {
  63. /* shortcut for most common case */
  64. if (sizeof(ISC_QUAD) == sizeof(ISC_UINT64)) {
  65. return sscanf(id, BLOB_ID_MASK, (ISC_UINT64 *) qd);
  66. } else {
  67. ISC_UINT64 res;
  68. if (sscanf(id, BLOB_ID_MASK, &res)) {
  69. qd->gds_quad_high = (ISC_LONG) (res >> 0x20);
  70. qd->gds_quad_low = (ISC_LONG) (res & 0xFFFFFFFF);
  71. return 1;
  72. }
  73. return 0;
  74. }
  75. }
  76. /* }}} */
  77. zend_string *_php_ibase_quad_to_string(ISC_QUAD const qd) /* {{{ */
  78. {
  79. /* shortcut for most common case */
  80. if (sizeof(ISC_QUAD) == sizeof(ISC_UINT64)) {
  81. return strpprintf(BLOB_ID_LEN+1, "0x%0*" LL_MASK "x", 16, *(ISC_UINT64*)(void *) &qd);
  82. } else {
  83. ISC_UINT64 res = ((ISC_UINT64) qd.gds_quad_high << 0x20) | qd.gds_quad_low;
  84. return strpprintf(BLOB_ID_LEN+1, "0x%0*" LL_MASK "x", 16, res);
  85. }
  86. }
  87. /* }}} */
  88. typedef struct { /* {{{ */
  89. ISC_LONG max_segment; /* Length of longest segment */
  90. ISC_LONG num_segments; /* Total number of segments */
  91. ISC_LONG total_length; /* Total length of blob */
  92. int bl_stream; /* blob is stream ? */
  93. /* }}} */
  94. } IBASE_BLOBINFO;
  95. int _php_ibase_blob_get(zval *return_value, ibase_blob *ib_blob, zend_ulong max_len) /* {{{ */
  96. {
  97. if (ib_blob->bl_qd.gds_quad_high || ib_blob->bl_qd.gds_quad_low) { /*not null ?*/
  98. ISC_STATUS stat;
  99. zend_string *bl_data;
  100. zend_ulong cur_len;
  101. unsigned short seg_len;
  102. bl_data = zend_string_safe_alloc(1, max_len, 0, 0);
  103. for (cur_len = stat = 0; (stat == 0 || stat == isc_segment) && cur_len < max_len; cur_len += seg_len) {
  104. unsigned short chunk_size = (max_len-cur_len) > USHRT_MAX ? USHRT_MAX
  105. : (unsigned short)(max_len-cur_len);
  106. stat = isc_get_segment(IB_STATUS, &ib_blob->bl_handle, &seg_len, chunk_size, &ZSTR_VAL(bl_data)[cur_len]);
  107. }
  108. if (IB_STATUS[0] == 1 && (stat != 0 && stat != isc_segstr_eof && stat != isc_segment)) {
  109. zend_string_free(bl_data);
  110. _php_ibase_error();
  111. return FAILURE;
  112. }
  113. ZSTR_VAL(bl_data)[cur_len] = '\0';
  114. ZSTR_LEN(bl_data) = cur_len;
  115. RETVAL_NEW_STR(bl_data);
  116. } else { /* null blob */
  117. RETVAL_EMPTY_STRING(); /* empty string */
  118. }
  119. return SUCCESS;
  120. }
  121. /* }}} */
  122. int _php_ibase_blob_add(zval *string_arg, ibase_blob *ib_blob) /* {{{ */
  123. {
  124. zend_ulong put_cnt = 0, rem_cnt;
  125. unsigned short chunk_size;
  126. convert_to_string_ex(string_arg);
  127. for (rem_cnt = Z_STRLEN_P(string_arg); rem_cnt > 0; rem_cnt -= chunk_size) {
  128. chunk_size = rem_cnt > USHRT_MAX ? USHRT_MAX : (unsigned short)rem_cnt;
  129. if (isc_put_segment(IB_STATUS, &ib_blob->bl_handle, chunk_size, &Z_STRVAL_P(string_arg)[put_cnt] )) {
  130. _php_ibase_error();
  131. return FAILURE;
  132. }
  133. put_cnt += chunk_size;
  134. }
  135. return SUCCESS;
  136. }
  137. /* }}} */
  138. static int _php_ibase_blob_info(isc_blob_handle bl_handle, IBASE_BLOBINFO *bl_info) /* {{{ */
  139. {
  140. static char bl_items[] = {
  141. isc_info_blob_num_segments,
  142. isc_info_blob_max_segment,
  143. isc_info_blob_total_length,
  144. isc_info_blob_type
  145. };
  146. char bl_inf[sizeof(zend_long)*8], *p;
  147. bl_info->max_segment = 0;
  148. bl_info->num_segments = 0;
  149. bl_info->total_length = 0;
  150. bl_info->bl_stream = 0;
  151. if (isc_blob_info(IB_STATUS, &bl_handle, sizeof(bl_items), bl_items, sizeof(bl_inf), bl_inf)) {
  152. _php_ibase_error();
  153. return FAILURE;
  154. }
  155. for (p = bl_inf; *p != isc_info_end && p < bl_inf + sizeof(bl_inf);) {
  156. unsigned short item_len;
  157. int item = *p++;
  158. item_len = (short) isc_vax_integer(p, 2);
  159. p += 2;
  160. switch (item) {
  161. case isc_info_blob_num_segments:
  162. bl_info->num_segments = isc_vax_integer(p, item_len);
  163. break;
  164. case isc_info_blob_max_segment:
  165. bl_info->max_segment = isc_vax_integer(p, item_len);
  166. break;
  167. case isc_info_blob_total_length:
  168. bl_info->total_length = isc_vax_integer(p, item_len);
  169. break;
  170. case isc_info_blob_type:
  171. bl_info->bl_stream = isc_vax_integer(p, item_len);
  172. break;
  173. case isc_info_end:
  174. break;
  175. case isc_info_truncated:
  176. case isc_info_error: /* hmm. don't think so...*/
  177. _php_ibase_module_error("PHP module internal error");
  178. return FAILURE;
  179. } /* switch */
  180. p += item_len;
  181. } /* for */
  182. return SUCCESS;
  183. }
  184. /* }}} */
  185. /* {{{ proto resource ibase_blob_create([resource link_identifier])
  186. Create blob for adding data */
  187. PHP_FUNCTION(ibase_blob_create)
  188. {
  189. zval *link = NULL;
  190. ibase_db_link *ib_link;
  191. ibase_trans *trans = NULL;
  192. ibase_blob *ib_blob;
  193. RESET_ERRMSG;
  194. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "|r", &link)) {
  195. RETURN_FALSE;
  196. }
  197. PHP_IBASE_LINK_TRANS(link, ib_link, trans);
  198. ib_blob = (ibase_blob *) emalloc(sizeof(ibase_blob));
  199. ib_blob->bl_handle = 0;
  200. ib_blob->type = BLOB_INPUT;
  201. if (isc_create_blob(IB_STATUS, &ib_link->handle, &trans->handle, &ib_blob->bl_handle, &ib_blob->bl_qd)) {
  202. _php_ibase_error();
  203. efree(ib_blob);
  204. RETURN_FALSE;
  205. }
  206. RETVAL_RES(zend_register_resource(ib_blob, le_blob));
  207. Z_TRY_ADDREF_P(return_value);
  208. }
  209. /* }}} */
  210. /* {{{ proto resource ibase_blob_open([ resource link_identifier, ] string blob_id)
  211. Open blob for retrieving data parts */
  212. PHP_FUNCTION(ibase_blob_open)
  213. {
  214. char *blob_id;
  215. size_t blob_id_len;
  216. zval *link = NULL;
  217. ibase_db_link *ib_link;
  218. ibase_trans *trans = NULL;
  219. ibase_blob *ib_blob;
  220. RESET_ERRMSG;
  221. PARSE_PARAMETERS;
  222. PHP_IBASE_LINK_TRANS(link, ib_link, trans);
  223. ib_blob = (ibase_blob *) emalloc(sizeof(ibase_blob));
  224. ib_blob->bl_handle = 0;
  225. ib_blob->type = BLOB_OUTPUT;
  226. do {
  227. if (! _php_ibase_string_to_quad(blob_id, &ib_blob->bl_qd)) {
  228. _php_ibase_module_error("String is not a BLOB ID");
  229. break;
  230. }
  231. if (isc_open_blob(IB_STATUS, &ib_link->handle, &trans->handle, &ib_blob->bl_handle,
  232. &ib_blob->bl_qd)) {
  233. _php_ibase_error();
  234. break;
  235. }
  236. RETVAL_RES(zend_register_resource(ib_blob, le_blob));
  237. Z_TRY_ADDREF_P(return_value);
  238. return;
  239. } while (0);
  240. efree(ib_blob);
  241. RETURN_FALSE;
  242. }
  243. /* }}} */
  244. /* {{{ proto bool ibase_blob_add(resource blob_handle, string data)
  245. Add data into created blob */
  246. PHP_FUNCTION(ibase_blob_add)
  247. {
  248. zval *blob_arg, *string_arg;
  249. ibase_blob *ib_blob;
  250. RESET_ERRMSG;
  251. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &blob_arg, &string_arg)) {
  252. return;
  253. }
  254. ib_blob = (ibase_blob *)zend_fetch_resource_ex(blob_arg, "Interbase blob", le_blob);
  255. if (ib_blob->type != BLOB_INPUT) {
  256. _php_ibase_module_error("BLOB is not open for input");
  257. RETURN_FALSE;
  258. }
  259. if (_php_ibase_blob_add(string_arg, ib_blob) != SUCCESS) {
  260. RETURN_FALSE;
  261. }
  262. }
  263. /* }}} */
  264. /* {{{ proto string ibase_blob_get(resource blob_handle, int len)
  265. Get len bytes data from open blob */
  266. PHP_FUNCTION(ibase_blob_get)
  267. {
  268. zval *blob_arg;
  269. zend_ulong len_arg;
  270. ibase_blob *ib_blob;
  271. RESET_ERRMSG;
  272. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "rl", &blob_arg, &len_arg)) {
  273. return;
  274. }
  275. ib_blob = (ibase_blob *)zend_fetch_resource_ex(blob_arg, "Interbase blob", le_blob);
  276. if (ib_blob->type != BLOB_OUTPUT) {
  277. _php_ibase_module_error("BLOB is not open for output");
  278. RETURN_FALSE;
  279. }
  280. if (_php_ibase_blob_get(return_value, ib_blob, len_arg) != SUCCESS) {
  281. RETURN_FALSE;
  282. }
  283. }
  284. /* }}} */
  285. static void _php_ibase_blob_end(INTERNAL_FUNCTION_PARAMETERS, int bl_end) /* {{{ */
  286. {
  287. zval *blob_arg;
  288. ibase_blob *ib_blob;
  289. RESET_ERRMSG;
  290. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "r", &blob_arg)) {
  291. return;
  292. }
  293. ib_blob = (ibase_blob *)zend_fetch_resource_ex(blob_arg, "Interbase blob", le_blob);
  294. if (bl_end == BLOB_CLOSE) { /* return id here */
  295. if (ib_blob->bl_qd.gds_quad_high || ib_blob->bl_qd.gds_quad_low) { /*not null ?*/
  296. if (isc_close_blob(IB_STATUS, &ib_blob->bl_handle)) {
  297. _php_ibase_error();
  298. RETURN_FALSE;
  299. }
  300. }
  301. ib_blob->bl_handle = 0;
  302. RETVAL_NEW_STR(_php_ibase_quad_to_string(ib_blob->bl_qd));
  303. } else { /* discard created blob */
  304. if (isc_cancel_blob(IB_STATUS, &ib_blob->bl_handle)) {
  305. _php_ibase_error();
  306. RETURN_FALSE;
  307. }
  308. ib_blob->bl_handle = 0;
  309. RETVAL_TRUE;
  310. }
  311. zend_list_delete(Z_RES_P(blob_arg));
  312. }
  313. /* }}} */
  314. /* {{{ proto string ibase_blob_close(resource blob_handle)
  315. Close blob */
  316. PHP_FUNCTION(ibase_blob_close)
  317. {
  318. _php_ibase_blob_end(INTERNAL_FUNCTION_PARAM_PASSTHRU, BLOB_CLOSE);
  319. }
  320. /* }}} */
  321. /* {{{ proto bool ibase_blob_cancel(resource blob_handle)
  322. Cancel creating blob */
  323. PHP_FUNCTION(ibase_blob_cancel)
  324. {
  325. _php_ibase_blob_end(INTERNAL_FUNCTION_PARAM_PASSTHRU, BLOB_CANCEL);
  326. }
  327. /* }}} */
  328. /* {{{ proto array ibase_blob_info([ resource link_identifier, ] string blob_id)
  329. Return blob length and other useful info */
  330. PHP_FUNCTION(ibase_blob_info)
  331. {
  332. char *blob_id;
  333. size_t blob_id_len;
  334. zval *link = NULL;
  335. ibase_db_link *ib_link;
  336. ibase_trans *trans = NULL;
  337. ibase_blob ib_blob = { 0, BLOB_INPUT };
  338. IBASE_BLOBINFO bl_info;
  339. RESET_ERRMSG;
  340. PARSE_PARAMETERS;
  341. PHP_IBASE_LINK_TRANS(link, ib_link, trans);
  342. if (! _php_ibase_string_to_quad(blob_id, &ib_blob.bl_qd)) {
  343. _php_ibase_module_error("Unrecognized BLOB ID");
  344. RETURN_FALSE;
  345. }
  346. if (ib_blob.bl_qd.gds_quad_high || ib_blob.bl_qd.gds_quad_low) { /* not null ? */
  347. if (isc_open_blob(IB_STATUS, &ib_link->handle, &trans->handle, &ib_blob.bl_handle,
  348. &ib_blob.bl_qd)) {
  349. _php_ibase_error();
  350. RETURN_FALSE;
  351. }
  352. if (_php_ibase_blob_info(ib_blob.bl_handle, &bl_info)) {
  353. RETURN_FALSE;
  354. }
  355. if (isc_close_blob(IB_STATUS, &ib_blob.bl_handle)) {
  356. _php_ibase_error();
  357. RETURN_FALSE;
  358. }
  359. } else { /* null blob, all values to zero */
  360. bl_info.max_segment = 0;
  361. bl_info.num_segments = 0;
  362. bl_info.total_length = 0;
  363. bl_info.bl_stream = 0;
  364. }
  365. array_init(return_value);
  366. add_index_long(return_value, 0, bl_info.total_length);
  367. add_assoc_long(return_value, "length", bl_info.total_length);
  368. add_index_long(return_value, 1, bl_info.num_segments);
  369. add_assoc_long(return_value, "numseg", bl_info.num_segments);
  370. add_index_long(return_value, 2, bl_info.max_segment);
  371. add_assoc_long(return_value, "maxseg", bl_info.max_segment);
  372. add_index_bool(return_value, 3, bl_info.bl_stream);
  373. add_assoc_bool(return_value, "stream", bl_info.bl_stream);
  374. add_index_bool(return_value, 4, (!ib_blob.bl_qd.gds_quad_high && !ib_blob.bl_qd.gds_quad_low));
  375. add_assoc_bool(return_value, "isnull", (!ib_blob.bl_qd.gds_quad_high && !ib_blob.bl_qd.gds_quad_low));
  376. }
  377. /* }}} */
  378. /* {{{ proto bool ibase_blob_echo([ resource link_identifier, ] string blob_id)
  379. Output blob contents to browser */
  380. PHP_FUNCTION(ibase_blob_echo)
  381. {
  382. char *blob_id;
  383. size_t blob_id_len;
  384. zval *link = NULL;
  385. ibase_db_link *ib_link;
  386. ibase_trans *trans = NULL;
  387. ibase_blob ib_blob_id = { 0, BLOB_OUTPUT };
  388. char bl_data[IBASE_BLOB_SEG];
  389. unsigned short seg_len;
  390. RESET_ERRMSG;
  391. PARSE_PARAMETERS;
  392. PHP_IBASE_LINK_TRANS(link, ib_link, trans);
  393. if (! _php_ibase_string_to_quad(blob_id, &ib_blob_id.bl_qd)) {
  394. _php_ibase_module_error("Unrecognized BLOB ID");
  395. RETURN_FALSE;
  396. }
  397. do {
  398. if (isc_open_blob(IB_STATUS, &ib_link->handle, &trans->handle, &ib_blob_id.bl_handle,
  399. &ib_blob_id.bl_qd)) {
  400. break;
  401. }
  402. while (!isc_get_segment(IB_STATUS, &ib_blob_id.bl_handle, &seg_len, sizeof(bl_data), bl_data)
  403. || IB_STATUS[1] == isc_segment) {
  404. PHPWRITE(bl_data, seg_len);
  405. }
  406. if (IB_STATUS[0] && (IB_STATUS[1] != isc_segstr_eof)) {
  407. break;
  408. }
  409. if (isc_close_blob(IB_STATUS, &ib_blob_id.bl_handle)) {
  410. break;
  411. }
  412. RETURN_TRUE;
  413. } while (0);
  414. _php_ibase_error();
  415. RETURN_FALSE;
  416. }
  417. /* }}} */
  418. /* {{{ proto string ibase_blob_import([ resource link_identifier, ] resource file)
  419. Create blob, copy file in it, and close it */
  420. PHP_FUNCTION(ibase_blob_import)
  421. {
  422. zval *link = NULL, *file;
  423. int size;
  424. unsigned short b;
  425. ibase_blob ib_blob = { 0, 0 };
  426. ibase_db_link *ib_link;
  427. ibase_trans *trans = NULL;
  428. char bl_data[IBASE_BLOB_SEG];
  429. php_stream *stream;
  430. RESET_ERRMSG;
  431. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "r|r",
  432. (ZEND_NUM_ARGS()-1) ? &link : &file, &file)) {
  433. RETURN_FALSE;
  434. }
  435. PHP_IBASE_LINK_TRANS(link, ib_link, trans);
  436. php_stream_from_zval(stream, file);
  437. do {
  438. if (isc_create_blob(IB_STATUS, &ib_link->handle, &trans->handle, &ib_blob.bl_handle,
  439. &ib_blob.bl_qd)) {
  440. break;
  441. }
  442. for (size = 0; (b = php_stream_read(stream, bl_data, sizeof(bl_data))); size += b) {
  443. if (isc_put_segment(IB_STATUS, &ib_blob.bl_handle, b, bl_data)) {
  444. break;
  445. }
  446. }
  447. if (isc_close_blob(IB_STATUS, &ib_blob.bl_handle)) {
  448. break;
  449. }
  450. RETURN_NEW_STR(_php_ibase_quad_to_string(ib_blob.bl_qd));
  451. } while (0);
  452. _php_ibase_error();
  453. RETURN_FALSE;
  454. }
  455. /* }}} */
  456. #endif /* HAVE_IBASE */
  457. /*
  458. * Local variables:
  459. * tab-width: 4
  460. * c-basic-offset: 4
  461. * End:
  462. * vim600: sw=4 ts=4 fdm=marker
  463. * vim<600: sw=4 ts=4
  464. */