com_persist.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767
  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. | Author: Wez Furlong <wez@thebrainroom.com> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* Infrastructure for working with persistent COM objects.
  19. * Implements: IStream* wrapper for PHP streams.
  20. * TODO: Magic __wakeup and __sleep handlers for serialization
  21. * (can wait till 5.1) */
  22. #ifdef HAVE_CONFIG_H
  23. #include "config.h"
  24. #endif
  25. #include "php.h"
  26. #include "php_ini.h"
  27. #include "ext/standard/info.h"
  28. #include "php_com_dotnet.h"
  29. #include "php_com_dotnet_internal.h"
  30. #include "Zend/zend_exceptions.h"
  31. /* {{{ expose php_stream as a COM IStream */
  32. typedef struct {
  33. CONST_VTBL struct IStreamVtbl *lpVtbl;
  34. DWORD engine_thread;
  35. LONG refcount;
  36. php_stream *stream;
  37. zend_resource *res;
  38. } php_istream;
  39. static int le_istream;
  40. static void istream_destructor(php_istream *stm);
  41. static void istream_dtor(zend_resource *rsrc)
  42. {
  43. php_istream *stm = (php_istream *)rsrc->ptr;
  44. istream_destructor(stm);
  45. }
  46. #define FETCH_STM() \
  47. php_istream *stm = (php_istream*)This; \
  48. if (GetCurrentThreadId() != stm->engine_thread) \
  49. return RPC_E_WRONG_THREAD;
  50. #define FETCH_STM_EX() \
  51. php_istream *stm = (php_istream*)This; \
  52. if (GetCurrentThreadId() != stm->engine_thread) \
  53. return RPC_E_WRONG_THREAD;
  54. static HRESULT STDMETHODCALLTYPE stm_queryinterface(
  55. IStream *This,
  56. /* [in] */ REFIID riid,
  57. /* [iid_is][out] */ void **ppvObject)
  58. {
  59. FETCH_STM_EX();
  60. if (IsEqualGUID(&IID_IUnknown, riid) ||
  61. IsEqualGUID(&IID_IStream, riid)) {
  62. *ppvObject = This;
  63. InterlockedIncrement(&stm->refcount);
  64. return S_OK;
  65. }
  66. *ppvObject = NULL;
  67. return E_NOINTERFACE;
  68. }
  69. static ULONG STDMETHODCALLTYPE stm_addref(IStream *This)
  70. {
  71. FETCH_STM_EX();
  72. return InterlockedIncrement(&stm->refcount);
  73. }
  74. static ULONG STDMETHODCALLTYPE stm_release(IStream *This)
  75. {
  76. ULONG ret;
  77. FETCH_STM();
  78. ret = InterlockedDecrement(&stm->refcount);
  79. if (ret == 0) {
  80. /* destroy it */
  81. if (stm->res)
  82. zend_list_delete(stm->res);
  83. }
  84. return ret;
  85. }
  86. static HRESULT STDMETHODCALLTYPE stm_read(IStream *This, void *pv, ULONG cb, ULONG *pcbRead)
  87. {
  88. ULONG nread;
  89. FETCH_STM();
  90. nread = (ULONG)php_stream_read(stm->stream, pv, cb);
  91. if (pcbRead) {
  92. *pcbRead = nread > 0 ? nread : 0;
  93. }
  94. if (nread > 0) {
  95. return S_OK;
  96. }
  97. return S_FALSE;
  98. }
  99. static HRESULT STDMETHODCALLTYPE stm_write(IStream *This, void const *pv, ULONG cb, ULONG *pcbWritten)
  100. {
  101. ULONG nwrote;
  102. FETCH_STM();
  103. nwrote = (ULONG)php_stream_write(stm->stream, pv, cb);
  104. if (pcbWritten) {
  105. *pcbWritten = nwrote > 0 ? nwrote : 0;
  106. }
  107. if (nwrote > 0) {
  108. return S_OK;
  109. }
  110. return S_FALSE;
  111. }
  112. static HRESULT STDMETHODCALLTYPE stm_seek(IStream *This, LARGE_INTEGER dlibMove,
  113. DWORD dwOrigin, ULARGE_INTEGER *plibNewPosition)
  114. {
  115. off_t offset;
  116. int whence;
  117. int ret;
  118. FETCH_STM();
  119. switch (dwOrigin) {
  120. case STREAM_SEEK_SET: whence = SEEK_SET; break;
  121. case STREAM_SEEK_CUR: whence = SEEK_CUR; break;
  122. case STREAM_SEEK_END: whence = SEEK_END; break;
  123. default:
  124. return STG_E_INVALIDFUNCTION;
  125. }
  126. if (dlibMove.HighPart) {
  127. /* we don't support 64-bit offsets */
  128. return STG_E_INVALIDFUNCTION;
  129. }
  130. offset = (off_t) dlibMove.QuadPart;
  131. ret = php_stream_seek(stm->stream, offset, whence);
  132. if (plibNewPosition) {
  133. plibNewPosition->QuadPart = (ULONGLONG)(ret >= 0 ? ret : 0);
  134. }
  135. return ret >= 0 ? S_OK : STG_E_INVALIDFUNCTION;
  136. }
  137. static HRESULT STDMETHODCALLTYPE stm_set_size(IStream *This, ULARGE_INTEGER libNewSize)
  138. {
  139. FETCH_STM();
  140. if (libNewSize.HighPart) {
  141. return STG_E_INVALIDFUNCTION;
  142. }
  143. if (php_stream_truncate_supported(stm->stream)) {
  144. int ret = php_stream_truncate_set_size(stm->stream, (size_t)libNewSize.QuadPart);
  145. if (ret == 0) {
  146. return S_OK;
  147. }
  148. }
  149. return STG_E_INVALIDFUNCTION;
  150. }
  151. static HRESULT STDMETHODCALLTYPE stm_copy_to(IStream *This, IStream *pstm, ULARGE_INTEGER cb,
  152. ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten)
  153. {
  154. FETCH_STM_EX();
  155. return E_NOTIMPL;
  156. }
  157. static HRESULT STDMETHODCALLTYPE stm_commit(IStream *This, DWORD grfCommitFlags)
  158. {
  159. FETCH_STM();
  160. php_stream_flush(stm->stream);
  161. return S_OK;
  162. }
  163. static HRESULT STDMETHODCALLTYPE stm_revert(IStream *This)
  164. {
  165. /* NOP */
  166. return S_OK;
  167. }
  168. static HRESULT STDMETHODCALLTYPE stm_lock_region(IStream *This,
  169. ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD lockType)
  170. {
  171. return STG_E_INVALIDFUNCTION;
  172. }
  173. static HRESULT STDMETHODCALLTYPE stm_unlock_region(IStream *This,
  174. ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD lockType)
  175. {
  176. return STG_E_INVALIDFUNCTION;
  177. }
  178. static HRESULT STDMETHODCALLTYPE stm_stat(IStream *This,
  179. STATSTG *pstatstg, DWORD grfStatFlag)
  180. {
  181. return STG_E_INVALIDFUNCTION;
  182. }
  183. static HRESULT STDMETHODCALLTYPE stm_clone(IStream *This, IStream **ppstm)
  184. {
  185. return STG_E_INVALIDFUNCTION;
  186. }
  187. static struct IStreamVtbl php_istream_vtbl = {
  188. stm_queryinterface,
  189. stm_addref,
  190. stm_release,
  191. stm_read,
  192. stm_write,
  193. stm_seek,
  194. stm_set_size,
  195. stm_copy_to,
  196. stm_commit,
  197. stm_revert,
  198. stm_lock_region,
  199. stm_unlock_region,
  200. stm_stat,
  201. stm_clone
  202. };
  203. static void istream_destructor(php_istream *stm)
  204. {
  205. if (stm->refcount > 0) {
  206. CoDisconnectObject((IUnknown*)stm, 0);
  207. }
  208. zend_list_delete(stm->stream->res);
  209. CoTaskMemFree(stm);
  210. }
  211. /* }}} */
  212. PHP_COM_DOTNET_API IStream *php_com_wrapper_export_stream(php_stream *stream)
  213. {
  214. php_istream *stm = (php_istream*)CoTaskMemAlloc(sizeof(*stm));
  215. if (stm == NULL)
  216. return NULL;
  217. memset(stm, 0, sizeof(*stm));
  218. stm->engine_thread = GetCurrentThreadId();
  219. stm->lpVtbl = &php_istream_vtbl;
  220. stm->refcount = 1;
  221. stm->stream = stream;
  222. GC_ADDREF(stream->res);
  223. stm->res = zend_register_resource(stm, le_istream);
  224. return (IStream*)stm;
  225. }
  226. #define CPH_ME(fname, arginfo) PHP_ME(com_persist, fname, arginfo, ZEND_ACC_PUBLIC)
  227. #define CPH_SME(fname, arginfo) PHP_ME(com_persist, fname, arginfo, ZEND_ACC_ALLOW_STATIC|ZEND_ACC_PUBLIC)
  228. #define CPH_METHOD(fname) static PHP_METHOD(com_persist, fname)
  229. #define CPH_FETCH() php_com_persist_helper *helper = (php_com_persist_helper*)Z_OBJ_P(getThis());
  230. #define CPH_NO_OBJ() if (helper->unk == NULL) { php_com_throw_exception(E_INVALIDARG, "No COM object is associated with this helper instance"); return; }
  231. typedef struct {
  232. zend_object std;
  233. long codepage;
  234. IUnknown *unk;
  235. IPersistStream *ips;
  236. IPersistStreamInit *ipsi;
  237. IPersistFile *ipf;
  238. } php_com_persist_helper;
  239. static zend_object_handlers helper_handlers;
  240. static zend_class_entry *helper_ce;
  241. static inline HRESULT get_persist_stream(php_com_persist_helper *helper)
  242. {
  243. if (!helper->ips && helper->unk) {
  244. return IUnknown_QueryInterface(helper->unk, &IID_IPersistStream, &helper->ips);
  245. }
  246. return helper->ips ? S_OK : E_NOTIMPL;
  247. }
  248. static inline HRESULT get_persist_stream_init(php_com_persist_helper *helper)
  249. {
  250. if (!helper->ipsi && helper->unk) {
  251. return IUnknown_QueryInterface(helper->unk, &IID_IPersistStreamInit, &helper->ipsi);
  252. }
  253. return helper->ipsi ? S_OK : E_NOTIMPL;
  254. }
  255. static inline HRESULT get_persist_file(php_com_persist_helper *helper)
  256. {
  257. if (!helper->ipf && helper->unk) {
  258. return IUnknown_QueryInterface(helper->unk, &IID_IPersistFile, &helper->ipf);
  259. }
  260. return helper->ipf ? S_OK : E_NOTIMPL;
  261. }
  262. /* {{{ proto string COMPersistHelper::GetCurFile()
  263. Determines the filename into which an object will be saved, or false if none is set, via IPersistFile::GetCurFile */
  264. CPH_METHOD(GetCurFileName)
  265. {
  266. HRESULT res;
  267. OLECHAR *olename = NULL;
  268. CPH_FETCH();
  269. CPH_NO_OBJ();
  270. res = get_persist_file(helper);
  271. if (helper->ipf) {
  272. res = IPersistFile_GetCurFile(helper->ipf, &olename);
  273. if (res == S_OK) {
  274. size_t len;
  275. char *str = php_com_olestring_to_string(olename,
  276. &len, helper->codepage);
  277. RETVAL_STRINGL(str, len);
  278. // TODO: avoid reallocarion???
  279. efree(str);
  280. CoTaskMemFree(olename);
  281. return;
  282. } else if (res == S_FALSE) {
  283. CoTaskMemFree(olename);
  284. RETURN_FALSE;
  285. }
  286. php_com_throw_exception(res, NULL);
  287. } else {
  288. php_com_throw_exception(res, NULL);
  289. }
  290. }
  291. /* }}} */
  292. /* {{{ proto bool COMPersistHelper::SaveToFile(string filename [, bool remember])
  293. Persist object data to file, via IPersistFile::Save */
  294. CPH_METHOD(SaveToFile)
  295. {
  296. HRESULT res;
  297. char *filename, *fullpath = NULL;
  298. size_t filename_len;
  299. zend_bool remember = TRUE;
  300. OLECHAR *olefilename = NULL;
  301. CPH_FETCH();
  302. CPH_NO_OBJ();
  303. res = get_persist_file(helper);
  304. if (helper->ipf) {
  305. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "p!|b",
  306. &filename, &filename_len, &remember)) {
  307. php_com_throw_exception(E_INVALIDARG, "Invalid arguments");
  308. return;
  309. }
  310. if (filename) {
  311. fullpath = expand_filepath(filename, NULL);
  312. if (!fullpath) {
  313. RETURN_FALSE;
  314. }
  315. if (php_check_open_basedir(fullpath)) {
  316. efree(fullpath);
  317. RETURN_FALSE;
  318. }
  319. olefilename = php_com_string_to_olestring(fullpath, strlen(fullpath), helper->codepage);
  320. efree(fullpath);
  321. }
  322. res = IPersistFile_Save(helper->ipf, olefilename, remember);
  323. if (SUCCEEDED(res)) {
  324. if (!olefilename) {
  325. res = IPersistFile_GetCurFile(helper->ipf, &olefilename);
  326. if (S_OK == res) {
  327. IPersistFile_SaveCompleted(helper->ipf, olefilename);
  328. CoTaskMemFree(olefilename);
  329. olefilename = NULL;
  330. }
  331. } else if (remember) {
  332. IPersistFile_SaveCompleted(helper->ipf, olefilename);
  333. }
  334. }
  335. if (olefilename) {
  336. efree(olefilename);
  337. }
  338. if (FAILED(res)) {
  339. php_com_throw_exception(res, NULL);
  340. }
  341. } else {
  342. php_com_throw_exception(res, NULL);
  343. }
  344. }
  345. /* }}} */
  346. /* {{{ proto bool COMPersistHelper::LoadFromFile(string filename [, int flags])
  347. Load object data from file, via IPersistFile::Load */
  348. CPH_METHOD(LoadFromFile)
  349. {
  350. HRESULT res;
  351. char *filename, *fullpath;
  352. size_t filename_len;
  353. zend_long flags = 0;
  354. OLECHAR *olefilename;
  355. CPH_FETCH();
  356. CPH_NO_OBJ();
  357. res = get_persist_file(helper);
  358. if (helper->ipf) {
  359. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "p|l",
  360. &filename, &filename_len, &flags)) {
  361. php_com_throw_exception(E_INVALIDARG, "Invalid arguments");
  362. return;
  363. }
  364. if (!(fullpath = expand_filepath(filename, NULL))) {
  365. RETURN_FALSE;
  366. }
  367. if (php_check_open_basedir(fullpath)) {
  368. efree(fullpath);
  369. RETURN_FALSE;
  370. }
  371. olefilename = php_com_string_to_olestring(fullpath, strlen(fullpath), helper->codepage);
  372. efree(fullpath);
  373. res = IPersistFile_Load(helper->ipf, olefilename, (DWORD)flags);
  374. efree(olefilename);
  375. if (FAILED(res)) {
  376. php_com_throw_exception(res, NULL);
  377. }
  378. } else {
  379. php_com_throw_exception(res, NULL);
  380. }
  381. }
  382. /* }}} */
  383. /* {{{ proto int COMPersistHelper::GetMaxStreamSize()
  384. Gets maximum stream size required to store the object data, via IPersistStream::GetSizeMax (or IPersistStreamInit::GetSizeMax) */
  385. CPH_METHOD(GetMaxStreamSize)
  386. {
  387. HRESULT res;
  388. ULARGE_INTEGER size;
  389. CPH_FETCH();
  390. CPH_NO_OBJ();
  391. res = get_persist_stream_init(helper);
  392. if (helper->ipsi) {
  393. res = IPersistStreamInit_GetSizeMax(helper->ipsi, &size);
  394. } else {
  395. res = get_persist_stream(helper);
  396. if (helper->ips) {
  397. res = IPersistStream_GetSizeMax(helper->ips, &size);
  398. } else {
  399. php_com_throw_exception(res, NULL);
  400. return;
  401. }
  402. }
  403. if (res != S_OK) {
  404. php_com_throw_exception(res, NULL);
  405. } else {
  406. /* TODO: handle 64 bit properly */
  407. RETURN_LONG((zend_long)size.QuadPart);
  408. }
  409. }
  410. /* }}} */
  411. /* {{{ proto int COMPersistHelper::InitNew()
  412. Initializes the object to a default state, via IPersistStreamInit::InitNew */
  413. CPH_METHOD(InitNew)
  414. {
  415. HRESULT res;
  416. CPH_FETCH();
  417. CPH_NO_OBJ();
  418. res = get_persist_stream_init(helper);
  419. if (helper->ipsi) {
  420. res = IPersistStreamInit_InitNew(helper->ipsi);
  421. if (res != S_OK) {
  422. php_com_throw_exception(res, NULL);
  423. } else {
  424. RETURN_TRUE;
  425. }
  426. } else {
  427. php_com_throw_exception(res, NULL);
  428. }
  429. }
  430. /* }}} */
  431. /* {{{ proto mixed COMPersistHelper::LoadFromStream(resource stream)
  432. Initializes an object from the stream where it was previously saved, via IPersistStream::Load or OleLoadFromStream */
  433. CPH_METHOD(LoadFromStream)
  434. {
  435. zval *zstm;
  436. php_stream *stream;
  437. IStream *stm = NULL;
  438. HRESULT res;
  439. CPH_FETCH();
  440. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "r", &zstm)) {
  441. php_com_throw_exception(E_INVALIDARG, "invalid arguments");
  442. return;
  443. }
  444. php_stream_from_zval_no_verify(stream, zstm);
  445. if (stream == NULL) {
  446. php_com_throw_exception(E_INVALIDARG, "expected a stream");
  447. return;
  448. }
  449. stm = php_com_wrapper_export_stream(stream);
  450. if (stm == NULL) {
  451. php_com_throw_exception(E_UNEXPECTED, "failed to wrap stream");
  452. return;
  453. }
  454. res = S_OK;
  455. RETVAL_TRUE;
  456. if (helper->unk == NULL) {
  457. IDispatch *disp = NULL;
  458. /* we need to create an object and load using OleLoadFromStream */
  459. res = OleLoadFromStream(stm, &IID_IDispatch, &disp);
  460. if (SUCCEEDED(res)) {
  461. php_com_wrap_dispatch(return_value, disp, COMG(code_page));
  462. }
  463. } else {
  464. res = get_persist_stream_init(helper);
  465. if (helper->ipsi) {
  466. res = IPersistStreamInit_Load(helper->ipsi, stm);
  467. } else {
  468. res = get_persist_stream(helper);
  469. if (helper->ips) {
  470. res = IPersistStreamInit_Load(helper->ipsi, stm);
  471. }
  472. }
  473. }
  474. IStream_Release(stm);
  475. if (FAILED(res)) {
  476. php_com_throw_exception(res, NULL);
  477. RETURN_NULL();
  478. }
  479. }
  480. /* }}} */
  481. /* {{{ proto int COMPersistHelper::SaveToStream(resource stream)
  482. Saves the object to a stream, via IPersistStream::Save */
  483. CPH_METHOD(SaveToStream)
  484. {
  485. zval *zstm;
  486. php_stream *stream;
  487. IStream *stm = NULL;
  488. HRESULT res;
  489. CPH_FETCH();
  490. CPH_NO_OBJ();
  491. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "r", &zstm)) {
  492. php_com_throw_exception(E_INVALIDARG, "invalid arguments");
  493. return;
  494. }
  495. php_stream_from_zval_no_verify(stream, zstm);
  496. if (stream == NULL) {
  497. php_com_throw_exception(E_INVALIDARG, "expected a stream");
  498. return;
  499. }
  500. stm = php_com_wrapper_export_stream(stream);
  501. if (stm == NULL) {
  502. php_com_throw_exception(E_UNEXPECTED, "failed to wrap stream");
  503. return;
  504. }
  505. res = get_persist_stream_init(helper);
  506. if (helper->ipsi) {
  507. res = IPersistStreamInit_Save(helper->ipsi, stm, TRUE);
  508. } else {
  509. res = get_persist_stream(helper);
  510. if (helper->ips) {
  511. res = IPersistStream_Save(helper->ips, stm, TRUE);
  512. }
  513. }
  514. IStream_Release(stm);
  515. if (FAILED(res)) {
  516. php_com_throw_exception(res, NULL);
  517. return;
  518. }
  519. RETURN_TRUE;
  520. }
  521. /* }}} */
  522. /* {{{ proto COMPersistHelper::__construct([object com_object])
  523. Creates a persistence helper object, usually associated with a com_object */
  524. CPH_METHOD(__construct)
  525. {
  526. php_com_dotnet_object *obj = NULL;
  527. zval *zobj = NULL;
  528. CPH_FETCH();
  529. if (FAILURE == zend_parse_parameters(ZEND_NUM_ARGS(), "|O!",
  530. &zobj, php_com_variant_class_entry)) {
  531. php_com_throw_exception(E_INVALIDARG, "invalid arguments");
  532. return;
  533. }
  534. if (!zobj) {
  535. return;
  536. }
  537. obj = CDNO_FETCH(zobj);
  538. if (V_VT(&obj->v) != VT_DISPATCH || V_DISPATCH(&obj->v) == NULL) {
  539. php_com_throw_exception(E_INVALIDARG, "parameter must represent an IDispatch COM object");
  540. return;
  541. }
  542. /* it is always safe to cast an interface to IUnknown */
  543. helper->unk = (IUnknown*)V_DISPATCH(&obj->v);
  544. IUnknown_AddRef(helper->unk);
  545. helper->codepage = obj->code_page;
  546. }
  547. /* }}} */
  548. static const zend_function_entry com_persist_helper_methods[] = {
  549. CPH_ME(__construct, NULL)
  550. CPH_ME(GetCurFileName, NULL)
  551. CPH_ME(SaveToFile, NULL)
  552. CPH_ME(LoadFromFile, NULL)
  553. CPH_ME(GetMaxStreamSize, NULL)
  554. CPH_ME(InitNew, NULL)
  555. CPH_ME(LoadFromStream, NULL)
  556. CPH_ME(SaveToStream, NULL)
  557. PHP_FE_END
  558. };
  559. static void helper_free_storage(zend_object *obj)
  560. {
  561. php_com_persist_helper *object = (php_com_persist_helper*)obj;
  562. if (object->ipf) {
  563. IPersistFile_Release(object->ipf);
  564. }
  565. if (object->ips) {
  566. IPersistStream_Release(object->ips);
  567. }
  568. if (object->ipsi) {
  569. IPersistStreamInit_Release(object->ipsi);
  570. }
  571. if (object->unk) {
  572. IUnknown_Release(object->unk);
  573. }
  574. zend_object_std_dtor(&object->std);
  575. }
  576. static zend_object* helper_clone(zval *obj)
  577. {
  578. php_com_persist_helper *clone, *object = (php_com_persist_helper*)Z_OBJ_P(obj);
  579. clone = emalloc(sizeof(*object));
  580. memcpy(clone, object, sizeof(*object));
  581. zend_object_std_init(&clone->std, object->std.ce);
  582. if (clone->ipf) {
  583. IPersistFile_AddRef(clone->ipf);
  584. }
  585. if (clone->ips) {
  586. IPersistStream_AddRef(clone->ips);
  587. }
  588. if (clone->ipsi) {
  589. IPersistStreamInit_AddRef(clone->ipsi);
  590. }
  591. if (clone->unk) {
  592. IUnknown_AddRef(clone->unk);
  593. }
  594. return (zend_object*)clone;
  595. }
  596. static zend_object* helper_new(zend_class_entry *ce)
  597. {
  598. php_com_persist_helper *helper;
  599. helper = emalloc(sizeof(*helper));
  600. memset(helper, 0, sizeof(*helper));
  601. zend_object_std_init(&helper->std, helper_ce);
  602. helper->std.handlers = &helper_handlers;
  603. return &helper->std;
  604. }
  605. int php_com_persist_minit(INIT_FUNC_ARGS)
  606. {
  607. zend_class_entry ce;
  608. memcpy(&helper_handlers, &std_object_handlers, sizeof(helper_handlers));
  609. helper_handlers.free_obj = helper_free_storage;
  610. helper_handlers.clone_obj = helper_clone;
  611. INIT_CLASS_ENTRY(ce, "COMPersistHelper", com_persist_helper_methods);
  612. ce.create_object = helper_new;
  613. helper_ce = zend_register_internal_class(&ce);
  614. helper_ce->ce_flags |= ZEND_ACC_FINAL;
  615. le_istream = zend_register_list_destructors_ex(istream_dtor,
  616. NULL, "com_dotnet_istream_wrapper", module_number);
  617. return SUCCESS;
  618. }
  619. /*
  620. * Local variables:
  621. * tab-width: 4
  622. * c-basic-offset: 4
  623. * End:
  624. * vim600: noet sw=4 ts=4 fdm=marker
  625. * vim<600: noet sw=4 ts=4
  626. */