lws-plat-unix.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728
  1. #include "private-libwebsockets.h"
  2. #include <pwd.h>
  3. #include <grp.h>
  4. #ifdef LWS_WITH_PLUGINS
  5. #include <dlfcn.h>
  6. #endif
  7. #include <dirent.h>
  8. /*
  9. * included from libwebsockets.c for unix builds
  10. */
  11. unsigned long long time_in_microseconds(void)
  12. {
  13. struct timeval tv;
  14. gettimeofday(&tv, NULL);
  15. return ((unsigned long long)tv.tv_sec * 1000000LL) + tv.tv_usec;
  16. }
  17. LWS_VISIBLE int
  18. lws_get_random(struct lws_context *context, void *buf, int len)
  19. {
  20. return read(context->fd_random, (char *)buf, len);
  21. }
  22. LWS_VISIBLE int
  23. lws_send_pipe_choked(struct lws *wsi)
  24. {
  25. struct lws_pollfd fds;
  26. /* treat the fact we got a truncated send pending as if we're choked */
  27. if (wsi->trunc_len)
  28. return 1;
  29. fds.fd = wsi->sock;
  30. fds.events = POLLOUT;
  31. fds.revents = 0;
  32. if (poll(&fds, 1, 0) != 1)
  33. return 1;
  34. if ((fds.revents & POLLOUT) == 0)
  35. return 1;
  36. /* okay to send another packet without blocking */
  37. return 0;
  38. }
  39. LWS_VISIBLE int
  40. lws_poll_listen_fd(struct lws_pollfd *fd)
  41. {
  42. return poll(fd, 1, 0);
  43. }
  44. LWS_VISIBLE void
  45. lws_cancel_service_pt(struct lws *wsi)
  46. {
  47. struct lws_context_per_thread *pt = &wsi->context->pt[(int)wsi->tsi];
  48. char buf = 0;
  49. if (write(pt->dummy_pipe_fds[1], &buf, sizeof(buf)) != 1)
  50. lwsl_err("Cannot write to dummy pipe");
  51. }
  52. LWS_VISIBLE void
  53. lws_cancel_service(struct lws_context *context)
  54. {
  55. struct lws_context_per_thread *pt = &context->pt[0];
  56. char buf = 0, m = context->count_threads;
  57. while (m--) {
  58. if (write(pt->dummy_pipe_fds[1], &buf, sizeof(buf)) != 1)
  59. lwsl_err("Cannot write to dummy pipe");
  60. pt++;
  61. }
  62. }
  63. LWS_VISIBLE void lwsl_emit_syslog(int level, const char *line)
  64. {
  65. int syslog_level = LOG_DEBUG;
  66. switch (level) {
  67. case LLL_ERR:
  68. syslog_level = LOG_ERR;
  69. break;
  70. case LLL_WARN:
  71. syslog_level = LOG_WARNING;
  72. break;
  73. case LLL_NOTICE:
  74. syslog_level = LOG_NOTICE;
  75. break;
  76. case LLL_INFO:
  77. syslog_level = LOG_INFO;
  78. break;
  79. }
  80. syslog(syslog_level, "%s", line);
  81. }
  82. LWS_VISIBLE LWS_EXTERN int
  83. lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
  84. {
  85. struct lws_context_per_thread *pt = &context->pt[tsi];
  86. int n = -1, m, c;
  87. char buf;
  88. /* stay dead once we are dead */
  89. if (!context || !context->vhost_list)
  90. return 1;
  91. if (timeout_ms < 0)
  92. goto faked_service;
  93. lws_libev_run(context, tsi);
  94. lws_libuv_run(context, tsi);
  95. if (!context->service_tid_detected) {
  96. struct lws _lws;
  97. memset(&_lws, 0, sizeof(_lws));
  98. _lws.context = context;
  99. context->service_tid_detected =
  100. context->vhost_list->protocols[0].callback(
  101. &_lws, LWS_CALLBACK_GET_THREAD_ID, NULL, NULL, 0);
  102. }
  103. context->service_tid = context->service_tid_detected;
  104. /*
  105. * is there anybody with pending stuff that needs service forcing?
  106. */
  107. if (!lws_service_adjust_timeout(context, 1, tsi)) {
  108. /* -1 timeout means just do forced service */
  109. lws_plat_service_tsi(context, -1, pt->tid);
  110. /* still somebody left who wants forced service? */
  111. if (!lws_service_adjust_timeout(context, 1, pt->tid))
  112. /* yes... come back again quickly */
  113. timeout_ms = 0;
  114. }
  115. n = poll(pt->fds, pt->fds_count, timeout_ms);
  116. #ifdef LWS_OPENSSL_SUPPORT
  117. if (!pt->rx_draining_ext_list &&
  118. !lws_ssl_anybody_has_buffered_read_tsi(context, tsi) && !n) {
  119. #else
  120. if (!pt->rx_draining_ext_list && !n) /* poll timeout */ {
  121. #endif
  122. lws_service_fd_tsi(context, NULL, tsi);
  123. return 0;
  124. }
  125. faked_service:
  126. m = lws_service_flag_pending(context, tsi);
  127. if (m)
  128. c = -1; /* unknown limit */
  129. else
  130. if (n < 0) {
  131. if (LWS_ERRNO != LWS_EINTR)
  132. return -1;
  133. return 0;
  134. } else
  135. c = n;
  136. /* any socket with events to service? */
  137. for (n = 0; n < pt->fds_count && c; n++) {
  138. if (!pt->fds[n].revents)
  139. continue;
  140. c--;
  141. if (pt->fds[n].fd == pt->dummy_pipe_fds[0]) {
  142. if (read(pt->fds[n].fd, &buf, 1) != 1)
  143. lwsl_err("Cannot read from dummy pipe.");
  144. continue;
  145. }
  146. m = lws_service_fd_tsi(context, &pt->fds[n], tsi);
  147. if (m < 0)
  148. return -1;
  149. /* if something closed, retry this slot */
  150. if (m)
  151. n--;
  152. }
  153. return 0;
  154. }
  155. LWS_VISIBLE int
  156. lws_plat_check_connection_error(struct lws *wsi)
  157. {
  158. return 0;
  159. }
  160. LWS_VISIBLE int
  161. lws_plat_service(struct lws_context *context, int timeout_ms)
  162. {
  163. return lws_plat_service_tsi(context, timeout_ms, 0);
  164. }
  165. LWS_VISIBLE int
  166. lws_plat_set_socket_options(struct lws_vhost *vhost, int fd)
  167. {
  168. int optval = 1;
  169. socklen_t optlen = sizeof(optval);
  170. #if defined(__APPLE__) || \
  171. defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
  172. defined(__NetBSD__) || \
  173. defined(__OpenBSD__)
  174. struct protoent *tcp_proto;
  175. #endif
  176. if (vhost->ka_time) {
  177. /* enable keepalive on this socket */
  178. optval = 1;
  179. if (setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE,
  180. (const void *)&optval, optlen) < 0)
  181. return 1;
  182. #if defined(__APPLE__) || \
  183. defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
  184. defined(__NetBSD__) || \
  185. defined(__CYGWIN__) || defined(__OpenBSD__) || defined (__sun)
  186. /*
  187. * didn't find a way to set these per-socket, need to
  188. * tune kernel systemwide values
  189. */
  190. #else
  191. /* set the keepalive conditions we want on it too */
  192. optval = vhost->ka_time;
  193. if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE,
  194. (const void *)&optval, optlen) < 0)
  195. return 1;
  196. optval = vhost->ka_interval;
  197. if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPINTVL,
  198. (const void *)&optval, optlen) < 0)
  199. return 1;
  200. optval = vhost->ka_probes;
  201. if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPCNT,
  202. (const void *)&optval, optlen) < 0)
  203. return 1;
  204. #endif
  205. }
  206. /* Disable Nagle */
  207. optval = 1;
  208. #if defined (__sun)
  209. if (setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
  210. return 1;
  211. #elif !defined(__APPLE__) && \
  212. !defined(__FreeBSD__) && !defined(__FreeBSD_kernel__) && \
  213. !defined(__NetBSD__) && \
  214. !defined(__OpenBSD__)
  215. if (setsockopt(fd, SOL_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
  216. return 1;
  217. #else
  218. tcp_proto = getprotobyname("TCP");
  219. if (setsockopt(fd, tcp_proto->p_proto, TCP_NODELAY, &optval, optlen) < 0)
  220. return 1;
  221. #endif
  222. /* We are nonblocking... */
  223. if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0)
  224. return 1;
  225. return 0;
  226. }
  227. LWS_VISIBLE void
  228. lws_plat_drop_app_privileges(struct lws_context_creation_info *info)
  229. {
  230. if (info->gid != -1)
  231. if (setgid(info->gid))
  232. lwsl_warn("setgid: %s\n", strerror(LWS_ERRNO));
  233. if (info->uid != -1) {
  234. struct passwd *p = getpwuid(info->uid);
  235. if (p) {
  236. initgroups(p->pw_name, info->gid);
  237. if (setuid(info->uid))
  238. lwsl_warn("setuid: %s\n", strerror(LWS_ERRNO));
  239. else
  240. lwsl_notice("Set privs to user '%s'\n", p->pw_name);
  241. } else
  242. lwsl_warn("getpwuid: unable to find uid %d", info->uid);
  243. }
  244. }
  245. #ifdef LWS_WITH_PLUGINS
  246. #if defined(LWS_USE_LIBUV) && UV_VERSION_MAJOR > 0
  247. /* libuv.c implements these in a cross-platform way */
  248. #else
  249. static int filter(const struct dirent *ent)
  250. {
  251. if (!strcmp(ent->d_name, ".") || !strcmp(ent->d_name, ".."))
  252. return 0;
  253. return 1;
  254. }
  255. LWS_VISIBLE int
  256. lws_plat_plugins_init(struct lws_context * context, const char * const *d)
  257. {
  258. struct lws_plugin_capability lcaps;
  259. struct lws_plugin *plugin;
  260. lws_plugin_init_func initfunc;
  261. struct dirent **namelist;
  262. int n, i, m, ret = 0;
  263. char path[256];
  264. void *l;
  265. lwsl_notice(" Plugins:\n");
  266. while (d && *d) {
  267. n = scandir(*d, &namelist, filter, alphasort);
  268. if (n < 0) {
  269. lwsl_err("Scandir on %s failed\n", *d);
  270. return 1;
  271. }
  272. for (i = 0; i < n; i++) {
  273. if (strlen(namelist[i]->d_name) < 7)
  274. goto inval;
  275. lwsl_notice(" %s\n", namelist[i]->d_name);
  276. lws_snprintf(path, sizeof(path) - 1, "%s/%s", *d,
  277. namelist[i]->d_name);
  278. l = dlopen(path, RTLD_NOW);
  279. if (!l) {
  280. lwsl_err("Error loading DSO: %s\n", dlerror());
  281. while (i++ < n)
  282. free(namelist[i]);
  283. goto bail;
  284. }
  285. /* we could open it, can we get his init function? */
  286. m = lws_snprintf(path, sizeof(path) - 1, "init_%s",
  287. namelist[i]->d_name + 3 /* snip lib... */);
  288. path[m - 3] = '\0'; /* snip the .so */
  289. initfunc = dlsym(l, path);
  290. if (!initfunc) {
  291. lwsl_err("Failed to get init on %s: %s",
  292. namelist[i]->d_name, dlerror());
  293. dlclose(l);
  294. }
  295. lcaps.api_magic = LWS_PLUGIN_API_MAGIC;
  296. m = initfunc(context, &lcaps);
  297. if (m) {
  298. lwsl_err("Initializing %s failed %d\n",
  299. namelist[i]->d_name, m);
  300. dlclose(l);
  301. goto skip;
  302. }
  303. plugin = lws_malloc(sizeof(*plugin));
  304. if (!plugin) {
  305. lwsl_err("OOM\n");
  306. goto bail;
  307. }
  308. plugin->list = context->plugin_list;
  309. context->plugin_list = plugin;
  310. strncpy(plugin->name, namelist[i]->d_name, sizeof(plugin->name) - 1);
  311. plugin->name[sizeof(plugin->name) - 1] = '\0';
  312. plugin->l = l;
  313. plugin->caps = lcaps;
  314. context->plugin_protocol_count += lcaps.count_protocols;
  315. context->plugin_extension_count += lcaps.count_extensions;
  316. free(namelist[i]);
  317. continue;
  318. skip:
  319. dlclose(l);
  320. inval:
  321. free(namelist[i]);
  322. }
  323. free(namelist);
  324. d++;
  325. }
  326. bail:
  327. free(namelist);
  328. return ret;
  329. }
  330. LWS_VISIBLE int
  331. lws_plat_plugins_destroy(struct lws_context * context)
  332. {
  333. struct lws_plugin *plugin = context->plugin_list, *p;
  334. lws_plugin_destroy_func func;
  335. char path[256];
  336. int m;
  337. if (!plugin)
  338. return 0;
  339. lwsl_notice("%s\n", __func__);
  340. while (plugin) {
  341. p = plugin;
  342. m = lws_snprintf(path, sizeof(path) - 1, "destroy_%s", plugin->name + 3);
  343. path[m - 3] = '\0';
  344. func = dlsym(plugin->l, path);
  345. if (!func) {
  346. lwsl_err("Failed to get destroy on %s: %s",
  347. plugin->name, dlerror());
  348. goto next;
  349. }
  350. m = func(context);
  351. if (m)
  352. lwsl_err("Initializing %s failed %d\n",
  353. plugin->name, m);
  354. next:
  355. dlclose(p->l);
  356. plugin = p->list;
  357. p->list = NULL;
  358. free(p);
  359. }
  360. context->plugin_list = NULL;
  361. return 0;
  362. }
  363. #endif
  364. #endif
  365. #if 0
  366. static void
  367. sigabrt_handler(int x)
  368. {
  369. printf("%s\n", __func__);
  370. //*(char *)0 = 0;
  371. }
  372. #endif
  373. LWS_VISIBLE int
  374. lws_plat_context_early_init(void)
  375. {
  376. signal(SIGPIPE, SIG_IGN);
  377. // signal(SIGABRT, sigabrt_handler);
  378. return 0;
  379. }
  380. LWS_VISIBLE void
  381. lws_plat_context_early_destroy(struct lws_context *context)
  382. {
  383. }
  384. LWS_VISIBLE void
  385. lws_plat_context_late_destroy(struct lws_context *context)
  386. {
  387. struct lws_context_per_thread *pt = &context->pt[0];
  388. int m = context->count_threads;
  389. #ifdef LWS_WITH_PLUGINS
  390. if (context->plugin_list)
  391. lws_plat_plugins_destroy(context);
  392. #endif
  393. if (context->lws_lookup)
  394. lws_free(context->lws_lookup);
  395. while (m--) {
  396. close(pt->dummy_pipe_fds[0]);
  397. close(pt->dummy_pipe_fds[1]);
  398. pt++;
  399. }
  400. close(context->fd_random);
  401. }
  402. /* cast a struct sockaddr_in6 * into addr for ipv6 */
  403. LWS_VISIBLE int
  404. lws_interface_to_sa(int ipv6, const char *ifname, struct sockaddr_in *addr,
  405. size_t addrlen)
  406. {
  407. int rc = -1;
  408. struct ifaddrs *ifr;
  409. struct ifaddrs *ifc;
  410. #ifdef LWS_USE_IPV6
  411. struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *)addr;
  412. #endif
  413. getifaddrs(&ifr);
  414. for (ifc = ifr; ifc != NULL && rc; ifc = ifc->ifa_next) {
  415. if (!ifc->ifa_addr)
  416. continue;
  417. lwsl_info(" interface %s vs %s\n", ifc->ifa_name, ifname);
  418. if (strcmp(ifc->ifa_name, ifname))
  419. continue;
  420. switch (ifc->ifa_addr->sa_family) {
  421. case AF_INET:
  422. #ifdef LWS_USE_IPV6
  423. if (ipv6) {
  424. /* map IPv4 to IPv6 */
  425. bzero((char *)&addr6->sin6_addr,
  426. sizeof(struct in6_addr));
  427. addr6->sin6_addr.s6_addr[10] = 0xff;
  428. addr6->sin6_addr.s6_addr[11] = 0xff;
  429. memcpy(&addr6->sin6_addr.s6_addr[12],
  430. &((struct sockaddr_in *)ifc->ifa_addr)->sin_addr,
  431. sizeof(struct in_addr));
  432. } else
  433. #endif
  434. memcpy(addr,
  435. (struct sockaddr_in *)ifc->ifa_addr,
  436. sizeof(struct sockaddr_in));
  437. break;
  438. #ifdef LWS_USE_IPV6
  439. case AF_INET6:
  440. memcpy(&addr6->sin6_addr,
  441. &((struct sockaddr_in6 *)ifc->ifa_addr)->sin6_addr,
  442. sizeof(struct in6_addr));
  443. break;
  444. #endif
  445. default:
  446. continue;
  447. }
  448. rc = 0;
  449. }
  450. freeifaddrs(ifr);
  451. if (rc == -1) {
  452. /* check if bind to IP address */
  453. #ifdef LWS_USE_IPV6
  454. if (inet_pton(AF_INET6, ifname, &addr6->sin6_addr) == 1)
  455. rc = 0;
  456. else
  457. #endif
  458. if (inet_pton(AF_INET, ifname, &addr->sin_addr) == 1)
  459. rc = 0;
  460. }
  461. return rc;
  462. }
  463. LWS_VISIBLE void
  464. lws_plat_insert_socket_into_fds(struct lws_context *context, struct lws *wsi)
  465. {
  466. struct lws_context_per_thread *pt = &context->pt[(int)wsi->tsi];
  467. lws_libev_io(wsi, LWS_EV_START | LWS_EV_READ);
  468. lws_libuv_io(wsi, LWS_EV_START | LWS_EV_READ);
  469. pt->fds[pt->fds_count++].revents = 0;
  470. }
  471. LWS_VISIBLE void
  472. lws_plat_delete_socket_from_fds(struct lws_context *context,
  473. struct lws *wsi, int m)
  474. {
  475. struct lws_context_per_thread *pt = &context->pt[(int)wsi->tsi];
  476. lws_libev_io(wsi, LWS_EV_STOP | LWS_EV_READ | LWS_EV_WRITE);
  477. lws_libuv_io(wsi, LWS_EV_STOP | LWS_EV_READ | LWS_EV_WRITE);
  478. pt->fds_count--;
  479. }
  480. LWS_VISIBLE void
  481. lws_plat_service_periodic(struct lws_context *context)
  482. {
  483. /* if our parent went down, don't linger around */
  484. if (context->started_with_parent &&
  485. kill(context->started_with_parent, 0) < 0)
  486. kill(getpid(), SIGTERM);
  487. }
  488. LWS_VISIBLE int
  489. lws_plat_change_pollfd(struct lws_context *context,
  490. struct lws *wsi, struct lws_pollfd *pfd)
  491. {
  492. return 0;
  493. }
  494. LWS_VISIBLE const char *
  495. lws_plat_inet_ntop(int af, const void *src, char *dst, int cnt)
  496. {
  497. return inet_ntop(af, src, dst, cnt);
  498. }
  499. static lws_filefd_type
  500. _lws_plat_file_open(struct lws *wsi, const char *filename,
  501. unsigned long *filelen, int flags)
  502. {
  503. struct stat stat_buf;
  504. int ret = open(filename, flags, 0664);
  505. if (ret < 0)
  506. return LWS_INVALID_FILE;
  507. if (fstat(ret, &stat_buf) < 0) {
  508. close(ret);
  509. return LWS_INVALID_FILE;
  510. }
  511. *filelen = stat_buf.st_size;
  512. return ret;
  513. }
  514. static int
  515. _lws_plat_file_close(struct lws *wsi, lws_filefd_type fd)
  516. {
  517. return close(fd);
  518. }
  519. unsigned long
  520. _lws_plat_file_seek_cur(struct lws *wsi, lws_filefd_type fd, long offset)
  521. {
  522. return lseek(fd, offset, SEEK_CUR);
  523. }
  524. static int
  525. _lws_plat_file_read(struct lws *wsi, lws_filefd_type fd, unsigned long *amount,
  526. unsigned char *buf, unsigned long len)
  527. {
  528. long n;
  529. n = read((int)fd, buf, len);
  530. if (n == -1) {
  531. *amount = 0;
  532. return -1;
  533. }
  534. *amount = n;
  535. return 0;
  536. }
  537. static int
  538. _lws_plat_file_write(struct lws *wsi, lws_filefd_type fd, unsigned long *amount,
  539. unsigned char *buf, unsigned long len)
  540. {
  541. long n;
  542. n = write((int)fd, buf, len);
  543. if (n == -1) {
  544. *amount = 0;
  545. return -1;
  546. }
  547. *amount = n;
  548. return 0;
  549. }
  550. LWS_VISIBLE int
  551. lws_plat_init(struct lws_context *context,
  552. struct lws_context_creation_info *info)
  553. {
  554. struct lws_context_per_thread *pt = &context->pt[0];
  555. int n = context->count_threads, fd;
  556. /* master context has the global fd lookup array */
  557. context->lws_lookup = lws_zalloc(sizeof(struct lws *) *
  558. context->max_fds);
  559. if (context->lws_lookup == NULL) {
  560. lwsl_err("OOM on lws_lookup array for %d connections\n",
  561. context->max_fds);
  562. return 1;
  563. }
  564. lwsl_notice(" mem: platform fd map: %5u bytes\n",
  565. sizeof(struct lws *) * context->max_fds);
  566. fd = open(SYSTEM_RANDOM_FILEPATH, O_RDONLY);
  567. context->fd_random = fd;
  568. if (context->fd_random < 0) {
  569. lwsl_err("Unable to open random device %s %d\n",
  570. SYSTEM_RANDOM_FILEPATH, context->fd_random);
  571. return 1;
  572. }
  573. if (!lws_libev_init_fd_table(context) &&
  574. !lws_libuv_init_fd_table(context)) {
  575. /* otherwise libev handled it instead */
  576. while (n--) {
  577. if (pipe(pt->dummy_pipe_fds)) {
  578. lwsl_err("Unable to create pipe\n");
  579. return 1;
  580. }
  581. /* use the read end of pipe as first item */
  582. pt->fds[0].fd = pt->dummy_pipe_fds[0];
  583. pt->fds[0].events = LWS_POLLIN;
  584. pt->fds[0].revents = 0;
  585. pt->fds_count = 1;
  586. pt++;
  587. }
  588. }
  589. context->fops.open = _lws_plat_file_open;
  590. context->fops.close = _lws_plat_file_close;
  591. context->fops.seek_cur = _lws_plat_file_seek_cur;
  592. context->fops.read = _lws_plat_file_read;
  593. context->fops.write = _lws_plat_file_write;
  594. #ifdef LWS_WITH_PLUGINS
  595. if (info->plugin_dirs)
  596. lws_plat_plugins_init(context, info->plugin_dirs);
  597. #endif
  598. return 0;
  599. }