tsrm_nw.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  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. | Authors: Venkat Raghavan S <rvenkat@novell.com> |
  16. | Anantha Kesari H Y <hyanantha@novell.com> |
  17. +----------------------------------------------------------------------+
  18. */
  19. /* $Id$ */
  20. #include <stdlib.h>
  21. #include <stdio.h>
  22. #include <fcntl.h>
  23. #include "TSRM.h"
  24. #ifdef NETWARE
  25. #ifdef USE_MKFIFO
  26. #include <sys/stat.h>
  27. #elif !defined(USE_PIPE_OPEN) /* NXFifoOpen */
  28. #include <nks/fsio.h>
  29. #endif
  30. #include <nks/vm.h>
  31. #include <nks/memory.h>
  32. #include <string.h>
  33. #include "mktemp.h"
  34. /* strtok() call in LibC is abending when used in a different address space
  35. * -- hence using PHP's version itself for now
  36. */
  37. #include "tsrm_strtok_r.h"
  38. #define tsrm_strtok_r(a,b,c) strtok((a),(b))
  39. #define WHITESPACE " \t"
  40. #define MAX_ARGS 10
  41. TSRM_API FILE* popen(const char *commandline, const char *type)
  42. {
  43. char *command = NULL, *argv[MAX_ARGS] = {'\0'}, **env = NULL;
  44. char *tempName = "sys:/php/temp/phpXXXXXX.tmp";
  45. char *filePath = NULL;
  46. char *ptr = NULL;
  47. int ptrLen = 0, argc = 0, i = 0, envCount = 0, err = 0;
  48. FILE *stream = NULL;
  49. #if defined(USE_PIPE_OPEN) || defined(USE_MKFIFO)
  50. int pipe_handle;
  51. int mode = O_RDONLY;
  52. #else
  53. NXHandle_t pipe_handle;
  54. NXMode_t mode = NX_O_RDONLY;
  55. #endif
  56. NXExecEnvSpec_t envSpec;
  57. NXNameSpec_t nameSpec;
  58. NXVmId_t newVM = 0;
  59. /* Check for validity of input parameters */
  60. if (!commandline || !type)
  61. return NULL;
  62. /* Get temporary file name */
  63. filePath = mktemp(tempName);
  64. if (!filePath)
  65. return NULL;
  66. /* Set pipe mode according to type -- for now allow only "r" or "w" */
  67. if (strcmp(type, "r") == 0)
  68. #if defined(USE_PIPE_OPEN) || defined(USE_MKFIFO)
  69. mode = O_RDONLY;
  70. #else
  71. mode = NX_O_RDONLY;
  72. #endif
  73. else if (strcmp(type, "w") == 0)
  74. #if defined(USE_PIPE_OPEN) || defined(USE_MKFIFO)
  75. mode = O_WRONLY;
  76. #else
  77. mode = NX_O_WRONLY;
  78. #endif
  79. else
  80. return NULL;
  81. #ifdef USE_PIPE_OPEN
  82. pipe_handle = pipe_open(filePath, mode);
  83. if (pipe_handle == -1)
  84. return NULL;
  85. #elif defined(USE_MKFIFO)
  86. pipe_handle = mkfifo(filePath, mode);
  87. if (pipe_handle == -1)
  88. return NULL;
  89. #else
  90. /* - NetWare doesn't require first parameter
  91. * - Allowing LibC to choose the buffer size for now
  92. */
  93. err = NXFifoOpen(0, filePath, mode, 0, &pipe_handle);
  94. if (err)
  95. return NULL;
  96. #endif
  97. /* Copy the environment variables in preparation for the spawn call */
  98. envCount = NXGetEnvCount() + 1; /* add one for NULL */
  99. env = (char **) NXMemAlloc(sizeof(char *) * envCount, 0);
  100. if (!env)
  101. return NULL;
  102. err = NXCopyEnv(env, envCount);
  103. if (err) {
  104. NXMemFree (env);
  105. return NULL;
  106. }
  107. /* Separate commandline string into words */
  108. ptr = tsrm_strtok_r((char*)commandline, WHITESPACE, NULL);
  109. ptrLen = strlen(ptr);
  110. command = (char*)malloc(ptrLen + 1);
  111. if (!command) {
  112. NXMemFree (env);
  113. return NULL;
  114. }
  115. strcpy (command, ptr);
  116. ptr = tsrm_strtok_r(NULL, WHITESPACE, NULL);
  117. while (ptr && (argc < MAX_ARGS)) {
  118. ptrLen = strlen(ptr);
  119. argv[argc] = (char*)malloc(ptrLen + 1);
  120. if (!argv[argc]) {
  121. NXMemFree (env);
  122. if (command)
  123. free (command);
  124. for (i = 0; i < argc; i++) {
  125. if (argv[i])
  126. free (argv[i]);
  127. }
  128. return NULL;
  129. }
  130. strcpy (argv[argc], ptr);
  131. argc++;
  132. ptr = tsrm_strtok_r(NULL, WHITESPACE, NULL);
  133. }
  134. /* Setup the execution environment and spawn new process */
  135. envSpec.esFlags = 0; /* Not used */
  136. envSpec.esArgc = argc;
  137. envSpec.esArgv = (void **) argv;
  138. envSpec.esEnv = (void **) env;
  139. /* envSpec.esStdin.ssType = */
  140. envSpec.esStdout.ssType = NX_OBJ_FIFO;
  141. envSpec.esStderr.ssType = NX_OBJ_FILE;
  142. /* 'ssHandle' is not a struct/union/class member */
  143. /*
  144. envSpec.esStdin.ssHandle =
  145. envSpec.esStdout.ssHandle =
  146. envSpec.esStderr.ssHandle = -1;
  147. */
  148. envSpec.esStdin.ssPathCtx = NULL;
  149. envSpec.esStdout.ssPathCtx = NULL;
  150. envSpec.esStderr.ssPathCtx = NULL;
  151. #if defined(USE_PIPE_OPEN) || defined(USE_MKFIFO)
  152. if (mode == O_RDONLY) {
  153. #else
  154. if (mode == NX_O_RDONLY) {
  155. #endif
  156. envSpec.esStdin.ssPath = filePath;
  157. envSpec.esStdout.ssPath = stdout;
  158. } else { /* Write Only */
  159. envSpec.esStdin.ssPath = stdin;
  160. envSpec.esStdout.ssPath = filePath;
  161. }
  162. envSpec.esStderr.ssPath = stdout;
  163. nameSpec.ssType = NX_OBJ_FIFO;
  164. /* nameSpec.ssHandle = 0; */ /* 'ssHandle' is not a struct/union/class member */
  165. nameSpec.ssPathCtx = NULL; /* Not used */
  166. nameSpec.ssPath = argv[0];
  167. err = NXVmSpawn(&nameSpec, &envSpec, 0, &newVM);
  168. if (!err)
  169. /* Get file pointer corresponding to the pipe (file) opened */
  170. stream = fdopen(pipe_handle, type);
  171. /* Clean-up */
  172. if (env)
  173. NXMemFree (env);
  174. if (pipe_handle)
  175. #if defined(USE_PIPE_OPEN) || defined(USE_MKFIFO)
  176. close(pipe_handle);
  177. #else
  178. NXClose(pipe_handle);
  179. #endif
  180. if (command)
  181. free (command);
  182. for (i = 0; i < argc; i++) {
  183. if (argv[i])
  184. free (argv[i]);
  185. }
  186. return stream;
  187. }
  188. TSRM_API int pclose(FILE* stream)
  189. {
  190. int err = 0;
  191. NXHandle_t fd = 0;
  192. /* Get the process associated with this pipe (file) handle and terminate it */
  193. fd = fileno(stream);
  194. NXClose (fd);
  195. err = fclose(stream);
  196. return err;
  197. }
  198. #endif /* NETWARE */