oci_statement.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 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. | Author: Wez Furlong <wez@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* $Id$ */
  19. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #include "php_ini.h"
  24. #include "ext/standard/info.h"
  25. #include "pdo/php_pdo.h"
  26. #include "pdo/php_pdo_driver.h"
  27. #include "php_pdo_oci.h"
  28. #include "php_pdo_oci_int.h"
  29. #include "Zend/zend_extensions.h"
  30. #define PDO_OCI_LOBMAXSIZE (4294967295UL) /* OCI_LOBMAXSIZE */
  31. #define STMT_CALL(name, params) \
  32. do { \
  33. S->last_err = name params; \
  34. S->last_err = _oci_error(S->err, stmt->dbh, stmt, #name, S->last_err, FALSE, __FILE__, __LINE__ TSRMLS_CC); \
  35. if (S->last_err) { \
  36. return 0; \
  37. } \
  38. } while(0)
  39. #define STMT_CALL_MSG(name, msg, params) \
  40. do { \
  41. S->last_err = name params; \
  42. S->last_err = _oci_error(S->err, stmt->dbh, stmt, #name ": " #msg, S->last_err, FALSE, __FILE__, __LINE__ TSRMLS_CC); \
  43. if (S->last_err) { \
  44. return 0; \
  45. } \
  46. } while(0)
  47. static php_stream *oci_create_lob_stream(pdo_stmt_t *stmt, OCILobLocator *lob TSRMLS_DC);
  48. static int oci_stmt_dtor(pdo_stmt_t *stmt TSRMLS_DC) /* {{{ */
  49. {
  50. pdo_oci_stmt *S = (pdo_oci_stmt*)stmt->driver_data;
  51. HashTable *BC = stmt->bound_columns;
  52. HashTable *BP = stmt->bound_params;
  53. int i;
  54. if (S->stmt) {
  55. /* cancel server side resources for the statement if we didn't
  56. * fetch it all */
  57. OCIStmtFetch(S->stmt, S->err, 0, OCI_FETCH_NEXT, OCI_DEFAULT);
  58. /* free the handle */
  59. OCIHandleFree(S->stmt, OCI_HTYPE_STMT);
  60. S->stmt = NULL;
  61. }
  62. if (S->err) {
  63. OCIHandleFree(S->err, OCI_HTYPE_ERROR);
  64. S->err = NULL;
  65. }
  66. /* need to ensure these go away now */
  67. if (BC) {
  68. zend_hash_destroy(BC);
  69. FREE_HASHTABLE(stmt->bound_columns);
  70. stmt->bound_columns = NULL;
  71. }
  72. if (BP) {
  73. zend_hash_destroy(BP);
  74. FREE_HASHTABLE(stmt->bound_params);
  75. stmt->bound_params = NULL;
  76. }
  77. if (S->einfo.errmsg) {
  78. pefree(S->einfo.errmsg, stmt->dbh->is_persistent);
  79. S->einfo.errmsg = NULL;
  80. }
  81. if (S->cols) {
  82. for (i = 0; i < stmt->column_count; i++) {
  83. if (S->cols[i].data) {
  84. switch (S->cols[i].dtype) {
  85. case SQLT_BLOB:
  86. case SQLT_CLOB:
  87. OCIDescriptorFree(S->cols[i].data, OCI_DTYPE_LOB);
  88. break;
  89. default:
  90. efree(S->cols[i].data);
  91. }
  92. }
  93. }
  94. efree(S->cols);
  95. S->cols = NULL;
  96. }
  97. efree(S);
  98. stmt->driver_data = NULL;
  99. return 1;
  100. } /* }}} */
  101. static int oci_stmt_execute(pdo_stmt_t *stmt TSRMLS_DC) /* {{{ */
  102. {
  103. pdo_oci_stmt *S = (pdo_oci_stmt*)stmt->driver_data;
  104. ub4 rowcount;
  105. b4 mode;
  106. if (!S->stmt_type) {
  107. STMT_CALL_MSG(OCIAttrGet, "OCI_ATTR_STMT_TYPE",
  108. (S->stmt, OCI_HTYPE_STMT, &S->stmt_type, 0, OCI_ATTR_STMT_TYPE, S->err));
  109. }
  110. if (stmt->executed) {
  111. /* ensure that we cancel the cursor from a previous fetch */
  112. OCIStmtFetch(S->stmt, S->err, 0, OCI_FETCH_NEXT, OCI_DEFAULT);
  113. }
  114. #ifdef OCI_STMT_SCROLLABLE_READONLY /* needed for oci8 ? */
  115. if (S->exec_type == OCI_STMT_SCROLLABLE_READONLY) {
  116. mode = OCI_STMT_SCROLLABLE_READONLY;
  117. } else
  118. #endif
  119. if (stmt->dbh->auto_commit && !stmt->dbh->in_txn) {
  120. mode = OCI_COMMIT_ON_SUCCESS;
  121. } else {
  122. mode = OCI_DEFAULT;
  123. }
  124. STMT_CALL(OCIStmtExecute, (S->H->svc, S->stmt, S->err,
  125. (S->stmt_type == OCI_STMT_SELECT && !S->have_blobs) ? 0 : 1, 0, NULL, NULL,
  126. mode));
  127. if (!stmt->executed) {
  128. ub4 colcount;
  129. /* do first-time-only definition of bind/mapping stuff */
  130. /* how many columns do we have ? */
  131. STMT_CALL_MSG(OCIAttrGet, "ATTR_PARAM_COUNT",
  132. (S->stmt, OCI_HTYPE_STMT, &colcount, 0, OCI_ATTR_PARAM_COUNT, S->err));
  133. stmt->column_count = (int)colcount;
  134. if (S->cols) {
  135. int i;
  136. for (i = 0; i < stmt->column_count; i++) {
  137. if (S->cols[i].data) {
  138. switch (S->cols[i].dtype) {
  139. case SQLT_BLOB:
  140. case SQLT_CLOB:
  141. /* do nothing */
  142. break;
  143. default:
  144. efree(S->cols[i].data);
  145. }
  146. }
  147. }
  148. efree(S->cols);
  149. }
  150. S->cols = ecalloc(colcount, sizeof(pdo_oci_column));
  151. }
  152. STMT_CALL_MSG(OCIAttrGet, "ATTR_ROW_COUNT",
  153. (S->stmt, OCI_HTYPE_STMT, &rowcount, 0, OCI_ATTR_ROW_COUNT, S->err));
  154. stmt->row_count = (long)rowcount;
  155. return 1;
  156. } /* }}} */
  157. static sb4 oci_bind_input_cb(dvoid *ctx, OCIBind *bindp, ub4 iter, ub4 index, dvoid **bufpp, ub4 *alenp, ub1 *piecep, dvoid **indpp) /* {{{ */
  158. {
  159. struct pdo_bound_param_data *param = (struct pdo_bound_param_data*)ctx;
  160. pdo_oci_bound_param *P = (pdo_oci_bound_param*)param->driver_data;
  161. TSRMLS_FETCH();
  162. if (!param || !param->parameter) {
  163. php_error_docref(NULL TSRMLS_CC, E_WARNING, "param is NULL in oci_bind_input_cb; this should not happen");
  164. return OCI_ERROR;
  165. }
  166. *indpp = &P->indicator;
  167. if (P->thing) {
  168. *bufpp = P->thing;
  169. *alenp = sizeof(void*);
  170. } else if (ZVAL_IS_NULL(param->parameter)) {
  171. /* insert a NULL value into the column */
  172. P->indicator = -1; /* NULL */
  173. *bufpp = 0;
  174. *alenp = -1;
  175. } else if (!P->thing) {
  176. /* regular string bind */
  177. convert_to_string(param->parameter);
  178. *bufpp = Z_STRVAL_P(param->parameter);
  179. *alenp = Z_STRLEN_P(param->parameter);
  180. }
  181. *piecep = OCI_ONE_PIECE;
  182. return OCI_CONTINUE;
  183. } /* }}} */
  184. static sb4 oci_bind_output_cb(dvoid *ctx, OCIBind *bindp, ub4 iter, ub4 index, dvoid **bufpp, ub4 **alenpp, ub1 *piecep, dvoid **indpp, ub2 **rcodepp) /* {{{ */
  185. {
  186. struct pdo_bound_param_data *param = (struct pdo_bound_param_data*)ctx;
  187. pdo_oci_bound_param *P = (pdo_oci_bound_param*)param->driver_data;
  188. TSRMLS_FETCH();
  189. if (!param || !param->parameter) {
  190. php_error_docref(NULL TSRMLS_CC, E_WARNING, "param is NULL in oci_bind_output_cb; this should not happen");
  191. return OCI_ERROR;
  192. }
  193. if (PDO_PARAM_TYPE(param->param_type) == PDO_PARAM_LOB) {
  194. P->actual_len = sizeof(OCILobLocator*);
  195. *bufpp = P->thing;
  196. *alenpp = &P->actual_len;
  197. *piecep = OCI_ONE_PIECE;
  198. *rcodepp = &P->retcode;
  199. *indpp = &P->indicator;
  200. return OCI_CONTINUE;
  201. }
  202. if (Z_TYPE_P(param->parameter) == IS_OBJECT || Z_TYPE_P(param->parameter) == IS_RESOURCE) {
  203. return OCI_CONTINUE;
  204. }
  205. convert_to_string(param->parameter);
  206. zval_dtor(param->parameter);
  207. Z_STRLEN_P(param->parameter) = param->max_value_len;
  208. Z_STRVAL_P(param->parameter) = ecalloc(1, Z_STRLEN_P(param->parameter)+1);
  209. P->used_for_output = 1;
  210. P->actual_len = Z_STRLEN_P(param->parameter);
  211. *alenpp = &P->actual_len;
  212. *bufpp = Z_STRVAL_P(param->parameter);
  213. *piecep = OCI_ONE_PIECE;
  214. *rcodepp = &P->retcode;
  215. *indpp = &P->indicator;
  216. return OCI_CONTINUE;
  217. } /* }}} */
  218. static int oci_stmt_param_hook(pdo_stmt_t *stmt, struct pdo_bound_param_data *param, enum pdo_param_event event_type TSRMLS_DC) /* {{{ */
  219. {
  220. pdo_oci_stmt *S = (pdo_oci_stmt*)stmt->driver_data;
  221. /* we're only interested in parameters for prepared SQL right now */
  222. if (param->is_param) {
  223. pdo_oci_bound_param *P;
  224. sb4 value_sz = -1;
  225. P = (pdo_oci_bound_param*)param->driver_data;
  226. switch (event_type) {
  227. case PDO_PARAM_EVT_FETCH_PRE:
  228. case PDO_PARAM_EVT_FETCH_POST:
  229. case PDO_PARAM_EVT_NORMALIZE:
  230. /* Do nothing */
  231. break;
  232. case PDO_PARAM_EVT_FREE:
  233. P = param->driver_data;
  234. if (P) {
  235. efree(P);
  236. }
  237. break;
  238. case PDO_PARAM_EVT_ALLOC:
  239. P = (pdo_oci_bound_param*)ecalloc(1, sizeof(pdo_oci_bound_param));
  240. param->driver_data = P;
  241. /* figure out what we're doing */
  242. switch (PDO_PARAM_TYPE(param->param_type)) {
  243. case PDO_PARAM_STMT:
  244. return 0;
  245. case PDO_PARAM_LOB:
  246. /* P->thing is now an OCILobLocator * */
  247. P->oci_type = SQLT_BLOB;
  248. value_sz = sizeof(OCILobLocator*);
  249. break;
  250. case PDO_PARAM_STR:
  251. default:
  252. P->oci_type = SQLT_CHR;
  253. value_sz = param->max_value_len;
  254. if (param->max_value_len == 0) {
  255. value_sz = 1332; /* maximum size before value is interpreted as a LONG value */
  256. }
  257. }
  258. if (param->name) {
  259. STMT_CALL(OCIBindByName, (S->stmt,
  260. &P->bind, S->err, (text*)param->name,
  261. param->namelen, 0, value_sz, P->oci_type,
  262. &P->indicator, 0, &P->retcode, 0, 0,
  263. OCI_DATA_AT_EXEC));
  264. } else {
  265. STMT_CALL(OCIBindByPos, (S->stmt,
  266. &P->bind, S->err, param->paramno+1,
  267. 0, value_sz, P->oci_type,
  268. &P->indicator, 0, &P->retcode, 0, 0,
  269. OCI_DATA_AT_EXEC));
  270. }
  271. STMT_CALL(OCIBindDynamic, (P->bind,
  272. S->err,
  273. param, oci_bind_input_cb,
  274. param, oci_bind_output_cb));
  275. return 1;
  276. case PDO_PARAM_EVT_EXEC_PRE:
  277. P->indicator = 0;
  278. P->used_for_output = 0;
  279. if (PDO_PARAM_TYPE(param->param_type) == PDO_PARAM_LOB) {
  280. ub4 empty = 0;
  281. STMT_CALL(OCIDescriptorAlloc, (S->H->env, &P->thing, OCI_DTYPE_LOB, 0, NULL));
  282. STMT_CALL(OCIAttrSet, (P->thing, OCI_DTYPE_LOB, &empty, 0, OCI_ATTR_LOBEMPTY, S->err));
  283. S->have_blobs = 1;
  284. }
  285. return 1;
  286. case PDO_PARAM_EVT_EXEC_POST:
  287. /* fixup stuff set in motion in oci_bind_output_cb */
  288. if (P->used_for_output) {
  289. if (P->indicator == -1) {
  290. /* set up a NULL value */
  291. if (Z_TYPE_P(param->parameter) == IS_STRING
  292. #if ZEND_EXTENSION_API_NO < 220040718
  293. && Z_STRVAL_P(param->parameter) != empty_string
  294. #endif
  295. ) {
  296. /* OCI likes to stick non-terminated strings in things */
  297. *Z_STRVAL_P(param->parameter) = '\0';
  298. }
  299. zval_dtor(param->parameter);
  300. ZVAL_NULL(param->parameter);
  301. } else if (Z_TYPE_P(param->parameter) == IS_STRING
  302. #if ZEND_EXTENSION_API_NO < 220040718
  303. && Z_STRVAL_P(param->parameter) != empty_string
  304. #endif
  305. ) {
  306. Z_STRLEN_P(param->parameter) = P->actual_len;
  307. Z_STRVAL_P(param->parameter) = erealloc(Z_STRVAL_P(param->parameter), P->actual_len+1);
  308. Z_STRVAL_P(param->parameter)[P->actual_len] = '\0';
  309. }
  310. } else if (PDO_PARAM_TYPE(param->param_type) == PDO_PARAM_LOB && P->thing) {
  311. php_stream *stm;
  312. if (Z_TYPE_P(param->parameter) == IS_NULL) {
  313. /* if the param is NULL, then we assume that they
  314. * wanted to bind a lob locator into it from the query
  315. * */
  316. stm = oci_create_lob_stream(stmt, (OCILobLocator*)P->thing TSRMLS_CC);
  317. if (stm) {
  318. OCILobOpen(S->H->svc, S->err, (OCILobLocator*)P->thing, OCI_LOB_READWRITE);
  319. php_stream_to_zval(stm, param->parameter);
  320. P->thing = NULL;
  321. }
  322. } else {
  323. /* we're a LOB being used for insert; transfer the data now */
  324. size_t n;
  325. ub4 amt, offset = 1;
  326. char *consume;
  327. php_stream_from_zval_no_verify(stm, &param->parameter);
  328. if (stm) {
  329. OCILobOpen(S->H->svc, S->err, (OCILobLocator*)P->thing, OCI_LOB_READWRITE);
  330. do {
  331. char buf[8192];
  332. n = php_stream_read(stm, buf, sizeof(buf));
  333. if ((int)n <= 0) {
  334. break;
  335. }
  336. consume = buf;
  337. do {
  338. amt = n;
  339. OCILobWrite(S->H->svc, S->err, (OCILobLocator*)P->thing,
  340. &amt, offset, consume, n,
  341. OCI_ONE_PIECE,
  342. NULL, NULL, 0, SQLCS_IMPLICIT);
  343. offset += amt;
  344. n -= amt;
  345. consume += amt;
  346. } while (n);
  347. } while (1);
  348. OCILobClose(S->H->svc, S->err, (OCILobLocator*)P->thing);
  349. OCILobFlushBuffer(S->H->svc, S->err, (OCILobLocator*)P->thing, 0);
  350. } else if (Z_TYPE_P(param->parameter) == IS_STRING) {
  351. /* stick the string into the LOB */
  352. consume = Z_STRVAL_P(param->parameter);
  353. n = Z_STRLEN_P(param->parameter);
  354. if (n) {
  355. OCILobOpen(S->H->svc, S->err, (OCILobLocator*)P->thing, OCI_LOB_READWRITE);
  356. while (n) {
  357. amt = n;
  358. OCILobWrite(S->H->svc, S->err, (OCILobLocator*)P->thing,
  359. &amt, offset, consume, n,
  360. OCI_ONE_PIECE,
  361. NULL, NULL, 0, SQLCS_IMPLICIT);
  362. consume += amt;
  363. n -= amt;
  364. }
  365. OCILobClose(S->H->svc, S->err, (OCILobLocator*)P->thing);
  366. }
  367. }
  368. OCIDescriptorFree(P->thing, OCI_DTYPE_LOB);
  369. P->thing = NULL;
  370. }
  371. }
  372. return 1;
  373. }
  374. }
  375. return 1;
  376. } /* }}} */
  377. static int oci_stmt_fetch(pdo_stmt_t *stmt, enum pdo_fetch_orientation ori, long offset TSRMLS_DC) /* {{{ */
  378. {
  379. #if HAVE_OCISTMTFETCH2
  380. ub4 ociori;
  381. #endif
  382. pdo_oci_stmt *S = (pdo_oci_stmt*)stmt->driver_data;
  383. #if HAVE_OCISTMTFETCH2
  384. switch (ori) {
  385. case PDO_FETCH_ORI_NEXT: ociori = OCI_FETCH_NEXT; break;
  386. case PDO_FETCH_ORI_PRIOR: ociori = OCI_FETCH_PRIOR; break;
  387. case PDO_FETCH_ORI_FIRST: ociori = OCI_FETCH_FIRST; break;
  388. case PDO_FETCH_ORI_LAST: ociori = OCI_FETCH_LAST; break;
  389. case PDO_FETCH_ORI_ABS: ociori = OCI_FETCH_ABSOLUTE; break;
  390. case PDO_FETCH_ORI_REL: ociori = OCI_FETCH_RELATIVE; break;
  391. }
  392. S->last_err = OCIStmtFetch2(S->stmt, S->err, 1, ociori, offset, OCI_DEFAULT);
  393. #else
  394. S->last_err = OCIStmtFetch(S->stmt, S->err, 1, OCI_FETCH_NEXT, OCI_DEFAULT);
  395. #endif
  396. if (S->last_err == OCI_NO_DATA) {
  397. /* no (more) data */
  398. return 0;
  399. }
  400. if (S->last_err == OCI_NEED_DATA) {
  401. oci_stmt_error("OCI_NEED_DATA");
  402. return 0;
  403. }
  404. if (S->last_err == OCI_SUCCESS_WITH_INFO || S->last_err == OCI_SUCCESS) {
  405. return 1;
  406. }
  407. oci_stmt_error("OCIStmtFetch");
  408. return 0;
  409. } /* }}} */
  410. static sb4 oci_define_callback(dvoid *octxp, OCIDefine *define, ub4 iter, dvoid **bufpp,
  411. ub4 **alenpp, ub1 *piecep, dvoid **indpp, ub2 **rcodepp)
  412. {
  413. pdo_oci_column *col = (pdo_oci_column*)octxp;
  414. TSRMLS_FETCH();
  415. switch (col->dtype) {
  416. case SQLT_BLOB:
  417. case SQLT_CLOB:
  418. *piecep = OCI_ONE_PIECE;
  419. *bufpp = col->data;
  420. *alenpp = &col->datalen;
  421. *indpp = (dvoid *)&col->indicator;
  422. break;
  423. default:
  424. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  425. "unhandled datatype in oci_define_callback; this should not happen");
  426. return OCI_ERROR;
  427. }
  428. return OCI_CONTINUE;
  429. }
  430. static int oci_stmt_describe(pdo_stmt_t *stmt, int colno TSRMLS_DC) /* {{{ */
  431. {
  432. pdo_oci_stmt *S = (pdo_oci_stmt*)stmt->driver_data;
  433. OCIParam *param = NULL;
  434. text *colname;
  435. ub2 dtype, data_size, scale, precis;
  436. ub4 namelen;
  437. struct pdo_column_data *col = &stmt->columns[colno];
  438. zend_bool dyn = FALSE;
  439. /* describe the column */
  440. STMT_CALL(OCIParamGet, (S->stmt, OCI_HTYPE_STMT, S->err, (dvoid*)&param, colno+1));
  441. /* what type ? */
  442. STMT_CALL_MSG(OCIAttrGet, "OCI_ATTR_DATA_TYPE",
  443. (param, OCI_DTYPE_PARAM, &dtype, 0, OCI_ATTR_DATA_TYPE, S->err));
  444. /* how big ? */
  445. STMT_CALL_MSG(OCIAttrGet, "OCI_ATTR_DATA_SIZE",
  446. (param, OCI_DTYPE_PARAM, &data_size, 0, OCI_ATTR_DATA_SIZE, S->err));
  447. /* scale ? */
  448. STMT_CALL_MSG(OCIAttrGet, "OCI_ATTR_SCALE",
  449. (param, OCI_DTYPE_PARAM, &scale, 0, OCI_ATTR_SCALE, S->err));
  450. /* precision ? */
  451. STMT_CALL_MSG(OCIAttrGet, "OCI_ATTR_PRECISION",
  452. (param, OCI_DTYPE_PARAM, &precis, 0, OCI_ATTR_PRECISION, S->err));
  453. /* name ? */
  454. STMT_CALL_MSG(OCIAttrGet, "OCI_ATTR_NAME",
  455. (param, OCI_DTYPE_PARAM, &colname, &namelen, OCI_ATTR_NAME, S->err));
  456. col->precision = scale;
  457. col->maxlen = data_size;
  458. col->namelen = namelen;
  459. col->name = estrndup((char *)colname, namelen);
  460. S->cols[colno].dtype = dtype;
  461. /* how much room do we need to store the field */
  462. switch (dtype) {
  463. case SQLT_LBI:
  464. case SQLT_LNG:
  465. if (dtype == SQLT_LBI) {
  466. dtype = SQLT_BIN;
  467. } else {
  468. dtype = SQLT_CHR;
  469. }
  470. S->cols[colno].datalen = 512; /* XXX should be INT_MAX and fetched by pieces */
  471. S->cols[colno].data = emalloc(S->cols[colno].datalen + 1);
  472. col->param_type = PDO_PARAM_STR;
  473. break;
  474. case SQLT_BLOB:
  475. case SQLT_CLOB:
  476. col->param_type = PDO_PARAM_LOB;
  477. STMT_CALL(OCIDescriptorAlloc, (S->H->env, (dvoid**)&S->cols[colno].data, OCI_DTYPE_LOB, 0, NULL));
  478. S->cols[colno].datalen = sizeof(OCILobLocator*);
  479. dyn = TRUE;
  480. break;
  481. case SQLT_BIN:
  482. default:
  483. if (dtype == SQLT_DAT || dtype == SQLT_NUM || dtype == SQLT_RDD
  484. #ifdef SQLT_TIMESTAMP
  485. || dtype == SQLT_TIMESTAMP
  486. #endif
  487. #ifdef SQLT_TIMESTAMP_TZ
  488. || dtype == SQLT_TIMESTAMP_TZ
  489. #endif
  490. ) {
  491. /* should be big enough for most date formats and numbers */
  492. S->cols[colno].datalen = 512;
  493. #if defined(SQLT_IBFLOAT) && defined(SQLT_IBDOUBLE)
  494. } else if (dtype == SQLT_IBFLOAT || dtype == SQLT_IBDOUBLE) {
  495. S->cols[colno].datalen = 1024;
  496. #endif
  497. } else {
  498. S->cols[colno].datalen = col->maxlen;
  499. }
  500. if (dtype == SQLT_BIN) {
  501. S->cols[colno].datalen *= 3;
  502. }
  503. S->cols[colno].data = emalloc(S->cols[colno].datalen + 1);
  504. dtype = SQLT_CHR;
  505. /* returning data as a string */
  506. col->param_type = PDO_PARAM_STR;
  507. }
  508. STMT_CALL(OCIDefineByPos, (S->stmt, &S->cols[colno].def, S->err, colno+1,
  509. S->cols[colno].data, S->cols[colno].datalen, dtype, &S->cols[colno].indicator,
  510. &S->cols[colno].fetched_len, &S->cols[colno].retcode, dyn ? OCI_DYNAMIC_FETCH : OCI_DEFAULT));
  511. if (dyn) {
  512. STMT_CALL(OCIDefineDynamic, (S->cols[colno].def, S->err, &S->cols[colno],
  513. oci_define_callback));
  514. }
  515. return 1;
  516. } /* }}} */
  517. struct oci_lob_self {
  518. pdo_stmt_t *stmt;
  519. pdo_oci_stmt *S;
  520. OCILobLocator *lob;
  521. ub4 offset;
  522. };
  523. static size_t oci_blob_write(php_stream *stream, const char *buf, size_t count TSRMLS_DC)
  524. {
  525. struct oci_lob_self *self = (struct oci_lob_self*)stream->abstract;
  526. ub4 amt;
  527. sword r;
  528. amt = count;
  529. r = OCILobWrite(self->S->H->svc, self->S->err, self->lob,
  530. &amt, self->offset, (char*)buf, count,
  531. OCI_ONE_PIECE,
  532. NULL, NULL, 0, SQLCS_IMPLICIT);
  533. if (r != OCI_SUCCESS) {
  534. return (size_t)-1;
  535. }
  536. self->offset += amt;
  537. return amt;
  538. }
  539. static size_t oci_blob_read(php_stream *stream, char *buf, size_t count TSRMLS_DC)
  540. {
  541. struct oci_lob_self *self = (struct oci_lob_self*)stream->abstract;
  542. ub4 amt;
  543. sword r;
  544. amt = count;
  545. r = OCILobRead(self->S->H->svc, self->S->err, self->lob,
  546. &amt, self->offset, buf, count,
  547. NULL, NULL, 0, SQLCS_IMPLICIT);
  548. if (r != OCI_SUCCESS && r != OCI_NEED_DATA) {
  549. return (size_t)-1;
  550. }
  551. self->offset += amt;
  552. if (amt < count) {
  553. stream->eof = 1;
  554. }
  555. return amt;
  556. }
  557. static int oci_blob_close(php_stream *stream, int close_handle TSRMLS_DC)
  558. {
  559. struct oci_lob_self *self = (struct oci_lob_self*)stream->abstract;
  560. pdo_stmt_t *stmt = self->stmt;
  561. if (close_handle) {
  562. OCILobClose(self->S->H->svc, self->S->err, self->lob);
  563. efree(self);
  564. }
  565. php_pdo_stmt_delref(stmt TSRMLS_CC);
  566. return 0;
  567. }
  568. static int oci_blob_flush(php_stream *stream TSRMLS_DC)
  569. {
  570. struct oci_lob_self *self = (struct oci_lob_self*)stream->abstract;
  571. OCILobFlushBuffer(self->S->H->svc, self->S->err, self->lob, 0);
  572. return 0;
  573. }
  574. static int oci_blob_seek(php_stream *stream, off_t offset, int whence, off_t *newoffset TSRMLS_DC)
  575. {
  576. struct oci_lob_self *self = (struct oci_lob_self*)stream->abstract;
  577. if (offset >= PDO_OCI_LOBMAXSIZE) {
  578. return -1;
  579. } else {
  580. self->offset = offset + 1; /* Oracle LOBS are 1-based, but PHP is 0-based */
  581. return 0;
  582. }
  583. }
  584. static php_stream_ops oci_blob_stream_ops = {
  585. oci_blob_write,
  586. oci_blob_read,
  587. oci_blob_close,
  588. oci_blob_flush,
  589. "pdo_oci blob stream",
  590. oci_blob_seek,
  591. NULL,
  592. NULL,
  593. NULL
  594. };
  595. static php_stream *oci_create_lob_stream(pdo_stmt_t *stmt, OCILobLocator *lob TSRMLS_DC)
  596. {
  597. php_stream *stm;
  598. struct oci_lob_self *self = ecalloc(1, sizeof(*self));
  599. self->lob = lob;
  600. self->offset = 1; /* 1-based */
  601. self->stmt = stmt;
  602. self->S = (pdo_oci_stmt*)stmt->driver_data;
  603. stm = php_stream_alloc(&oci_blob_stream_ops, self, 0, "r+b");
  604. if (stm) {
  605. php_pdo_stmt_addref(stmt TSRMLS_CC);
  606. return stm;
  607. }
  608. efree(self);
  609. return NULL;
  610. }
  611. static int oci_stmt_get_col(pdo_stmt_t *stmt, int colno, char **ptr, unsigned long *len, int *caller_frees TSRMLS_DC) /* {{{ */
  612. {
  613. pdo_oci_stmt *S = (pdo_oci_stmt*)stmt->driver_data;
  614. pdo_oci_column *C = &S->cols[colno];
  615. /* check the indicator to ensure that the data is intact */
  616. if (C->indicator == -1) {
  617. /* A NULL value */
  618. *ptr = NULL;
  619. *len = 0;
  620. return 1;
  621. } else if (C->indicator == 0) {
  622. /* it was stored perfectly */
  623. if (C->dtype == SQLT_BLOB || C->dtype == SQLT_CLOB) {
  624. if (C->data) {
  625. *ptr = (char*)oci_create_lob_stream(stmt, (OCILobLocator*)C->data TSRMLS_CC);
  626. OCILobOpen(S->H->svc, S->err, (OCILobLocator*)C->data, OCI_LOB_READONLY);
  627. }
  628. *len = 0;
  629. return *ptr ? 1 : 0;
  630. }
  631. *ptr = C->data;
  632. *len = C->fetched_len;
  633. return 1;
  634. } else {
  635. /* it was truncated */
  636. php_error_docref(NULL TSRMLS_CC, E_WARNING, "column %d data was too large for buffer and was truncated to fit it", colno);
  637. *ptr = C->data;
  638. *len = C->fetched_len;
  639. return 1;
  640. }
  641. } /* }}} */
  642. struct pdo_stmt_methods oci_stmt_methods = {
  643. oci_stmt_dtor,
  644. oci_stmt_execute,
  645. oci_stmt_fetch,
  646. oci_stmt_describe,
  647. oci_stmt_get_col,
  648. oci_stmt_param_hook
  649. };
  650. /*
  651. * Local variables:
  652. * tab-width: 4
  653. * c-basic-offset: 4
  654. * End:
  655. * vim600: noet sw=4 ts=4 fdm=marker
  656. * vim<600: noet sw=4 ts=4
  657. */