2014-04-02 21:02:54 +08:00
|
|
|
#include "private-libwebsockets.h"
|
|
|
|
|
2015-03-04 16:16:41 +08:00
|
|
|
#include <pwd.h>
|
|
|
|
#include <grp.h>
|
|
|
|
|
2016-06-27 05:53:38 +08:00
|
|
|
#ifdef LWS_WITH_PLUGINS
|
2016-04-06 16:15:40 +08:00
|
|
|
#include <dlfcn.h>
|
2016-06-27 05:53:38 +08:00
|
|
|
#endif
|
2016-04-06 16:15:40 +08:00
|
|
|
#include <dirent.h>
|
|
|
|
|
|
|
|
|
2014-03-31 11:01:32 +08:00
|
|
|
/*
|
|
|
|
* included from libwebsockets.c for unix builds
|
|
|
|
*/
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
unsigned long long time_in_microseconds(void)
|
2014-03-31 11:01:32 +08:00
|
|
|
{
|
|
|
|
struct timeval tv;
|
|
|
|
gettimeofday(&tv, NULL);
|
2015-01-26 15:39:36 +08:00
|
|
|
return ((unsigned long long)tv.tv_sec * 1000000LL) + tv.tv_usec;
|
2014-03-31 11:01:32 +08:00
|
|
|
}
|
|
|
|
|
2015-12-17 17:03:59 +08:00
|
|
|
LWS_VISIBLE int
|
|
|
|
lws_get_random(struct lws_context *context, void *buf, int len)
|
2014-03-31 11:01:32 +08:00
|
|
|
{
|
|
|
|
return read(context->fd_random, (char *)buf, len);
|
|
|
|
}
|
|
|
|
|
2015-12-17 17:03:59 +08:00
|
|
|
LWS_VISIBLE int
|
|
|
|
lws_send_pipe_choked(struct lws *wsi)
|
2014-03-31 11:01:32 +08:00
|
|
|
{
|
2015-12-04 11:08:32 +08:00
|
|
|
struct lws_pollfd fds;
|
2014-03-31 11:01:32 +08:00
|
|
|
|
|
|
|
/* treat the fact we got a truncated send pending as if we're choked */
|
2015-12-17 17:03:59 +08:00
|
|
|
if (wsi->trunc_len)
|
2014-03-31 11:01:32 +08:00
|
|
|
return 1;
|
|
|
|
|
2017-02-27 12:55:56 +08:00
|
|
|
fds.fd = wsi->desc.sockfd;
|
2014-03-31 11:01:32 +08:00
|
|
|
fds.events = POLLOUT;
|
|
|
|
fds.revents = 0;
|
|
|
|
|
|
|
|
if (poll(&fds, 1, 0) != 1)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if ((fds.revents & POLLOUT) == 0)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* okay to send another packet without blocking */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE int
|
2015-12-04 11:08:32 +08:00
|
|
|
lws_poll_listen_fd(struct lws_pollfd *fd)
|
2014-03-31 11:01:32 +08:00
|
|
|
{
|
|
|
|
return poll(fd, 1, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
LWS_VISIBLE void
|
2016-01-19 03:34:24 +08:00
|
|
|
lws_cancel_service_pt(struct lws *wsi)
|
2014-03-31 11:01:32 +08:00
|
|
|
{
|
2016-01-19 03:34:24 +08:00
|
|
|
struct lws_context_per_thread *pt = &wsi->context->pt[(int)wsi->tsi];
|
2014-03-31 11:01:32 +08:00
|
|
|
char buf = 0;
|
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
if (write(pt->dummy_pipe_fds[1], &buf, sizeof(buf)) != 1)
|
2014-03-31 11:01:32 +08:00
|
|
|
lwsl_err("Cannot write to dummy pipe");
|
|
|
|
}
|
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
LWS_VISIBLE void
|
|
|
|
lws_cancel_service(struct lws_context *context)
|
|
|
|
{
|
|
|
|
struct lws_context_per_thread *pt = &context->pt[0];
|
|
|
|
char buf = 0, m = context->count_threads;
|
|
|
|
|
|
|
|
while (m--) {
|
|
|
|
if (write(pt->dummy_pipe_fds[1], &buf, sizeof(buf)) != 1)
|
|
|
|
lwsl_err("Cannot write to dummy pipe");
|
|
|
|
pt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-31 11:01:32 +08:00
|
|
|
LWS_VISIBLE void lwsl_emit_syslog(int level, const char *line)
|
|
|
|
{
|
|
|
|
int syslog_level = LOG_DEBUG;
|
|
|
|
|
|
|
|
switch (level) {
|
|
|
|
case LLL_ERR:
|
|
|
|
syslog_level = LOG_ERR;
|
|
|
|
break;
|
|
|
|
case LLL_WARN:
|
|
|
|
syslog_level = LOG_WARNING;
|
|
|
|
break;
|
|
|
|
case LLL_NOTICE:
|
|
|
|
syslog_level = LOG_NOTICE;
|
|
|
|
break;
|
|
|
|
case LLL_INFO:
|
|
|
|
syslog_level = LOG_INFO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
syslog(syslog_level, "%s", line);
|
2014-04-02 14:25:10 +08:00
|
|
|
}
|
|
|
|
|
2016-10-07 03:19:17 +08:00
|
|
|
LWS_VISIBLE LWS_EXTERN int
|
2016-10-20 09:09:56 +08:00
|
|
|
_lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
2016-12-24 07:57:34 +08:00
|
|
|
struct lws_context_per_thread *pt;
|
2016-04-06 09:23:16 +08:00
|
|
|
int n = -1, m, c;
|
2016-01-19 03:34:24 +08:00
|
|
|
char buf;
|
2014-04-02 14:25:10 +08:00
|
|
|
|
|
|
|
/* stay dead once we are dead */
|
|
|
|
|
2016-03-28 10:10:43 +08:00
|
|
|
if (!context || !context->vhost_list)
|
2014-04-02 14:25:10 +08:00
|
|
|
return 1;
|
|
|
|
|
2016-12-24 07:57:34 +08:00
|
|
|
pt = &context->pt[tsi];
|
|
|
|
|
2016-04-06 09:23:16 +08:00
|
|
|
if (timeout_ms < 0)
|
|
|
|
goto faked_service;
|
|
|
|
|
2016-02-14 09:27:41 +08:00
|
|
|
lws_libev_run(context, tsi);
|
|
|
|
lws_libuv_run(context, tsi);
|
2014-04-11 13:14:37 +08:00
|
|
|
|
2015-12-24 13:00:54 +08:00
|
|
|
if (!context->service_tid_detected) {
|
|
|
|
struct lws _lws;
|
|
|
|
|
|
|
|
memset(&_lws, 0, sizeof(_lws));
|
|
|
|
_lws.context = context;
|
|
|
|
|
2016-04-06 09:23:16 +08:00
|
|
|
context->service_tid_detected =
|
|
|
|
context->vhost_list->protocols[0].callback(
|
2015-12-24 13:00:54 +08:00
|
|
|
&_lws, LWS_CALLBACK_GET_THREAD_ID, NULL, NULL, 0);
|
|
|
|
}
|
|
|
|
context->service_tid = context->service_tid_detected;
|
2014-04-02 14:25:10 +08:00
|
|
|
|
2016-10-04 08:39:14 +08:00
|
|
|
/*
|
|
|
|
* is there anybody with pending stuff that needs service forcing?
|
|
|
|
*/
|
2016-10-10 20:34:34 +08:00
|
|
|
if (!lws_service_adjust_timeout(context, 1, tsi)) {
|
2016-10-04 08:39:14 +08:00
|
|
|
/* -1 timeout means just do forced service */
|
2016-10-20 09:09:56 +08:00
|
|
|
_lws_plat_service_tsi(context, -1, pt->tid);
|
2016-10-04 08:39:14 +08:00
|
|
|
/* still somebody left who wants forced service? */
|
2016-10-10 20:34:34 +08:00
|
|
|
if (!lws_service_adjust_timeout(context, 1, pt->tid))
|
|
|
|
/* yes... come back again quickly */
|
|
|
|
timeout_ms = 0;
|
2016-10-04 08:39:14 +08:00
|
|
|
}
|
2016-01-11 11:34:01 +08:00
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
n = poll(pt->fds, pt->fds_count, timeout_ms);
|
2014-04-02 14:25:10 +08:00
|
|
|
|
2014-10-09 08:29:22 +08:00
|
|
|
#ifdef LWS_OPENSSL_SUPPORT
|
2016-01-19 03:34:24 +08:00
|
|
|
if (!pt->rx_draining_ext_list &&
|
|
|
|
!lws_ssl_anybody_has_buffered_read_tsi(context, tsi) && !n) {
|
2014-10-09 08:29:22 +08:00
|
|
|
#else
|
2016-01-19 03:34:24 +08:00
|
|
|
if (!pt->rx_draining_ext_list && !n) /* poll timeout */ {
|
2014-10-09 08:29:22 +08:00
|
|
|
#endif
|
2016-01-19 03:34:24 +08:00
|
|
|
lws_service_fd_tsi(context, NULL, tsi);
|
2014-04-02 14:25:10 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-04-06 09:23:16 +08:00
|
|
|
faked_service:
|
2016-02-26 10:48:51 +08:00
|
|
|
m = lws_service_flag_pending(context, tsi);
|
|
|
|
if (m)
|
|
|
|
c = -1; /* unknown limit */
|
|
|
|
else
|
|
|
|
if (n < 0) {
|
|
|
|
if (LWS_ERRNO != LWS_EINTR)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
} else
|
|
|
|
c = n;
|
2015-01-29 08:36:18 +08:00
|
|
|
|
|
|
|
/* any socket with events to service? */
|
2016-01-26 20:56:56 +08:00
|
|
|
for (n = 0; n < pt->fds_count && c; n++) {
|
2016-01-19 03:34:24 +08:00
|
|
|
if (!pt->fds[n].revents)
|
2014-04-02 14:25:10 +08:00
|
|
|
continue;
|
|
|
|
|
2016-01-26 20:56:56 +08:00
|
|
|
c--;
|
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
if (pt->fds[n].fd == pt->dummy_pipe_fds[0]) {
|
|
|
|
if (read(pt->fds[n].fd, &buf, 1) != 1)
|
2014-04-02 14:25:10 +08:00
|
|
|
lwsl_err("Cannot read from dummy pipe.");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
m = lws_service_fd_tsi(context, &pt->fds[n], tsi);
|
2014-04-02 14:25:10 +08:00
|
|
|
if (m < 0)
|
|
|
|
return -1;
|
|
|
|
/* if something closed, retry this slot */
|
|
|
|
if (m)
|
|
|
|
n--;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-06-28 19:01:20 +08:00
|
|
|
LWS_VISIBLE int
|
|
|
|
lws_plat_check_connection_error(struct lws *wsi)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
LWS_VISIBLE int
|
|
|
|
lws_plat_service(struct lws_context *context, int timeout_ms)
|
|
|
|
{
|
2016-10-20 09:09:56 +08:00
|
|
|
return _lws_plat_service_tsi(context, timeout_ms, 0);
|
2016-01-19 03:34:24 +08:00
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE int
|
2016-03-28 10:10:43 +08:00
|
|
|
lws_plat_set_socket_options(struct lws_vhost *vhost, int fd)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
|
|
|
int optval = 1;
|
|
|
|
socklen_t optlen = sizeof(optval);
|
|
|
|
|
2015-12-25 13:35:40 +02:00
|
|
|
#if defined(__APPLE__) || \
|
|
|
|
defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
|
|
|
|
defined(__NetBSD__) || \
|
2014-05-28 04:52:18 +00:00
|
|
|
defined(__OpenBSD__)
|
2014-04-02 14:25:10 +08:00
|
|
|
struct protoent *tcp_proto;
|
|
|
|
#endif
|
|
|
|
|
2016-03-28 10:10:43 +08:00
|
|
|
if (vhost->ka_time) {
|
2014-04-02 14:25:10 +08:00
|
|
|
/* enable keepalive on this socket */
|
|
|
|
optval = 1;
|
|
|
|
if (setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE,
|
2015-12-17 17:03:59 +08:00
|
|
|
(const void *)&optval, optlen) < 0)
|
2014-04-02 14:25:10 +08:00
|
|
|
return 1;
|
|
|
|
|
2015-12-25 13:35:40 +02:00
|
|
|
#if defined(__APPLE__) || \
|
|
|
|
defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
|
|
|
|
defined(__NetBSD__) || \
|
2016-09-09 06:49:44 +08:00
|
|
|
defined(__CYGWIN__) || defined(__OpenBSD__) || defined (__sun)
|
2014-04-02 14:25:10 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* didn't find a way to set these per-socket, need to
|
|
|
|
* tune kernel systemwide values
|
|
|
|
*/
|
|
|
|
#else
|
|
|
|
/* set the keepalive conditions we want on it too */
|
2016-03-28 10:10:43 +08:00
|
|
|
optval = vhost->ka_time;
|
2014-07-29 15:36:06 +03:00
|
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE,
|
2015-12-17 17:03:59 +08:00
|
|
|
(const void *)&optval, optlen) < 0)
|
2014-04-02 14:25:10 +08:00
|
|
|
return 1;
|
|
|
|
|
2016-03-28 10:10:43 +08:00
|
|
|
optval = vhost->ka_interval;
|
2014-07-29 15:36:06 +03:00
|
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPINTVL,
|
2015-12-17 17:03:59 +08:00
|
|
|
(const void *)&optval, optlen) < 0)
|
2014-04-02 14:25:10 +08:00
|
|
|
return 1;
|
|
|
|
|
2016-03-28 10:10:43 +08:00
|
|
|
optval = vhost->ka_probes;
|
2014-07-29 15:36:06 +03:00
|
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPCNT,
|
2015-12-17 17:03:59 +08:00
|
|
|
(const void *)&optval, optlen) < 0)
|
2014-04-02 14:25:10 +08:00
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable Nagle */
|
|
|
|
optval = 1;
|
2016-09-09 06:49:44 +08:00
|
|
|
#if defined (__sun)
|
|
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
|
|
|
|
return 1;
|
|
|
|
#elif !defined(__APPLE__) && \
|
|
|
|
!defined(__FreeBSD__) && !defined(__FreeBSD_kernel__) && \
|
|
|
|
!defined(__NetBSD__) && \
|
|
|
|
!defined(__OpenBSD__)
|
2014-11-30 13:06:09 +08:00
|
|
|
if (setsockopt(fd, SOL_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
|
|
|
|
return 1;
|
2014-04-02 14:25:10 +08:00
|
|
|
#else
|
|
|
|
tcp_proto = getprotobyname("TCP");
|
2014-11-30 13:06:09 +08:00
|
|
|
if (setsockopt(fd, tcp_proto->p_proto, TCP_NODELAY, &optval, optlen) < 0)
|
|
|
|
return 1;
|
2014-04-02 14:25:10 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* We are nonblocking... */
|
2014-11-30 13:02:32 +08:00
|
|
|
if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0)
|
|
|
|
return 1;
|
2014-04-02 14:25:10 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE void
|
|
|
|
lws_plat_drop_app_privileges(struct lws_context_creation_info *info)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
2016-02-25 15:12:16 +08:00
|
|
|
if (info->gid != -1)
|
|
|
|
if (setgid(info->gid))
|
|
|
|
lwsl_warn("setgid: %s\n", strerror(LWS_ERRNO));
|
|
|
|
|
2015-03-04 16:16:41 +08:00
|
|
|
if (info->uid != -1) {
|
|
|
|
struct passwd *p = getpwuid(info->uid);
|
|
|
|
|
|
|
|
if (p) {
|
|
|
|
initgroups(p->pw_name, info->gid);
|
|
|
|
if (setuid(info->uid))
|
|
|
|
lwsl_warn("setuid: %s\n", strerror(LWS_ERRNO));
|
|
|
|
else
|
2016-03-28 10:10:43 +08:00
|
|
|
lwsl_notice("Set privs to user '%s'\n", p->pw_name);
|
2015-03-04 16:16:41 +08:00
|
|
|
} else
|
|
|
|
lwsl_warn("getpwuid: unable to find uid %d", info->uid);
|
|
|
|
}
|
2014-04-02 14:25:10 +08:00
|
|
|
}
|
|
|
|
|
2016-04-06 16:15:40 +08:00
|
|
|
#ifdef LWS_WITH_PLUGINS
|
|
|
|
|
2016-04-09 07:22:40 +08:00
|
|
|
#if defined(LWS_USE_LIBUV) && UV_VERSION_MAJOR > 0
|
|
|
|
|
|
|
|
/* libuv.c implements these in a cross-platform way */
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2016-04-06 16:15:40 +08:00
|
|
|
static int filter(const struct dirent *ent)
|
|
|
|
{
|
|
|
|
if (!strcmp(ent->d_name, ".") || !strcmp(ent->d_name, ".."))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
LWS_VISIBLE int
|
2016-05-02 10:03:25 +08:00
|
|
|
lws_plat_plugins_init(struct lws_context * context, const char * const *d)
|
2016-04-06 16:15:40 +08:00
|
|
|
{
|
|
|
|
struct lws_plugin_capability lcaps;
|
|
|
|
struct lws_plugin *plugin;
|
|
|
|
lws_plugin_init_func initfunc;
|
|
|
|
struct dirent **namelist;
|
|
|
|
int n, i, m, ret = 0;
|
|
|
|
char path[256];
|
|
|
|
void *l;
|
|
|
|
|
2016-05-02 10:03:25 +08:00
|
|
|
lwsl_notice(" Plugins:\n");
|
2016-04-06 16:15:40 +08:00
|
|
|
|
2016-05-02 10:03:25 +08:00
|
|
|
while (d && *d) {
|
|
|
|
n = scandir(*d, &namelist, filter, alphasort);
|
|
|
|
if (n < 0) {
|
|
|
|
lwsl_err("Scandir on %s failed\n", *d);
|
|
|
|
return 1;
|
|
|
|
}
|
2016-04-06 16:15:40 +08:00
|
|
|
|
2016-05-02 10:03:25 +08:00
|
|
|
for (i = 0; i < n; i++) {
|
|
|
|
if (strlen(namelist[i]->d_name) < 7)
|
|
|
|
goto inval;
|
2016-04-06 16:15:40 +08:00
|
|
|
|
2016-05-02 10:03:25 +08:00
|
|
|
lwsl_notice(" %s\n", namelist[i]->d_name);
|
2016-04-06 16:15:40 +08:00
|
|
|
|
2016-09-15 02:22:57 +08:00
|
|
|
lws_snprintf(path, sizeof(path) - 1, "%s/%s", *d,
|
2016-05-02 10:03:25 +08:00
|
|
|
namelist[i]->d_name);
|
|
|
|
l = dlopen(path, RTLD_NOW);
|
|
|
|
if (!l) {
|
|
|
|
lwsl_err("Error loading DSO: %s\n", dlerror());
|
|
|
|
while (i++ < n)
|
|
|
|
free(namelist[i]);
|
|
|
|
goto bail;
|
|
|
|
}
|
|
|
|
/* we could open it, can we get his init function? */
|
2016-09-15 02:22:57 +08:00
|
|
|
m = lws_snprintf(path, sizeof(path) - 1, "init_%s",
|
2016-05-02 10:03:25 +08:00
|
|
|
namelist[i]->d_name + 3 /* snip lib... */);
|
|
|
|
path[m - 3] = '\0'; /* snip the .so */
|
|
|
|
initfunc = dlsym(l, path);
|
|
|
|
if (!initfunc) {
|
|
|
|
lwsl_err("Failed to get init on %s: %s",
|
|
|
|
namelist[i]->d_name, dlerror());
|
|
|
|
dlclose(l);
|
|
|
|
}
|
|
|
|
lcaps.api_magic = LWS_PLUGIN_API_MAGIC;
|
|
|
|
m = initfunc(context, &lcaps);
|
|
|
|
if (m) {
|
|
|
|
lwsl_err("Initializing %s failed %d\n",
|
|
|
|
namelist[i]->d_name, m);
|
|
|
|
dlclose(l);
|
|
|
|
goto skip;
|
|
|
|
}
|
2016-04-06 16:15:40 +08:00
|
|
|
|
2016-05-02 10:03:25 +08:00
|
|
|
plugin = lws_malloc(sizeof(*plugin));
|
|
|
|
if (!plugin) {
|
|
|
|
lwsl_err("OOM\n");
|
|
|
|
goto bail;
|
|
|
|
}
|
|
|
|
plugin->list = context->plugin_list;
|
|
|
|
context->plugin_list = plugin;
|
|
|
|
strncpy(plugin->name, namelist[i]->d_name, sizeof(plugin->name) - 1);
|
|
|
|
plugin->name[sizeof(plugin->name) - 1] = '\0';
|
|
|
|
plugin->l = l;
|
|
|
|
plugin->caps = lcaps;
|
|
|
|
context->plugin_protocol_count += lcaps.count_protocols;
|
|
|
|
context->plugin_extension_count += lcaps.count_extensions;
|
|
|
|
|
|
|
|
free(namelist[i]);
|
|
|
|
continue;
|
2016-04-06 16:15:40 +08:00
|
|
|
|
2016-05-02 10:03:25 +08:00
|
|
|
skip:
|
|
|
|
dlclose(l);
|
|
|
|
inval:
|
|
|
|
free(namelist[i]);
|
2016-04-06 16:15:40 +08:00
|
|
|
}
|
2016-05-02 10:03:25 +08:00
|
|
|
free(namelist);
|
|
|
|
d++;
|
2016-04-06 16:15:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bail:
|
|
|
|
free(namelist);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
LWS_VISIBLE int
|
|
|
|
lws_plat_plugins_destroy(struct lws_context * context)
|
|
|
|
{
|
|
|
|
struct lws_plugin *plugin = context->plugin_list, *p;
|
|
|
|
lws_plugin_destroy_func func;
|
|
|
|
char path[256];
|
|
|
|
int m;
|
|
|
|
|
|
|
|
if (!plugin)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
lwsl_notice("%s\n", __func__);
|
|
|
|
|
|
|
|
while (plugin) {
|
|
|
|
p = plugin;
|
2016-09-15 02:22:57 +08:00
|
|
|
m = lws_snprintf(path, sizeof(path) - 1, "destroy_%s", plugin->name + 3);
|
2016-04-06 16:15:40 +08:00
|
|
|
path[m - 3] = '\0';
|
|
|
|
func = dlsym(plugin->l, path);
|
|
|
|
if (!func) {
|
|
|
|
lwsl_err("Failed to get destroy on %s: %s",
|
|
|
|
plugin->name, dlerror());
|
|
|
|
goto next;
|
|
|
|
}
|
|
|
|
m = func(context);
|
|
|
|
if (m)
|
|
|
|
lwsl_err("Initializing %s failed %d\n",
|
|
|
|
plugin->name, m);
|
|
|
|
next:
|
|
|
|
dlclose(p->l);
|
|
|
|
plugin = p->list;
|
|
|
|
p->list = NULL;
|
|
|
|
free(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
context->plugin_list = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2016-04-09 07:22:40 +08:00
|
|
|
#endif
|
2016-04-06 16:15:40 +08:00
|
|
|
|
|
|
|
|
2016-04-10 09:33:54 +08:00
|
|
|
#if 0
|
2016-01-29 21:18:54 +08:00
|
|
|
static void
|
2016-04-13 11:49:07 +08:00
|
|
|
sigabrt_handler(int x)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
2016-04-13 11:49:07 +08:00
|
|
|
printf("%s\n", __func__);
|
|
|
|
//*(char *)0 = 0;
|
2014-04-02 14:25:10 +08:00
|
|
|
}
|
2016-04-10 09:33:54 +08:00
|
|
|
#endif
|
2014-04-02 14:25:10 +08:00
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE int
|
|
|
|
lws_plat_context_early_init(void)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
2016-04-13 11:49:07 +08:00
|
|
|
signal(SIGPIPE, SIG_IGN);
|
|
|
|
|
2016-04-10 09:33:54 +08:00
|
|
|
// signal(SIGABRT, sigabrt_handler);
|
2014-04-02 14:25:10 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE void
|
2015-12-04 11:08:32 +08:00
|
|
|
lws_plat_context_early_destroy(struct lws_context *context)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE void
|
2015-12-04 11:08:32 +08:00
|
|
|
lws_plat_context_late_destroy(struct lws_context *context)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
2016-01-19 03:34:24 +08:00
|
|
|
struct lws_context_per_thread *pt = &context->pt[0];
|
|
|
|
int m = context->count_threads;
|
|
|
|
|
2016-04-06 16:15:40 +08:00
|
|
|
#ifdef LWS_WITH_PLUGINS
|
|
|
|
if (context->plugin_list)
|
|
|
|
lws_plat_plugins_destroy(context);
|
|
|
|
#endif
|
|
|
|
|
2015-06-25 17:51:07 +02:00
|
|
|
if (context->lws_lookup)
|
|
|
|
lws_free(context->lws_lookup);
|
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
while (m--) {
|
context deprecation
1) This makes lwsws run a parent process with the original permissions.
But this process is only able to respond to SIGHUP, it doesn't do anything
else.
2) You can send this parent process a SIGHUP now to cause it to
- close listening sockets in existing lwsws processes
- mark those processes as to exit when the number of active connections
on the falls to zero
- spawn a fresh child process from scratch, using latest configuration
file content, latest plugins, etc. It can now reopen listening sockets
if it chooses to, or open different listen ports or whatever.
Notes:
1) lws_context_destroy() has been split into two pieces... the reason for
the split is the first part closes the per-vhost protocols, but since
they may have created libuv objects in the per-vhost protocol storage,
these cannot be freed until after the loop has been run.
That's the purpose of the second part of the context destruction,
lws_context_destroy2().
For compatibility, if you are not using libuv, the first part calls the
second part. However if you are using libuv, you must now call the
second part from your own main.c after the first part.
2016-12-16 07:37:43 +08:00
|
|
|
if (pt->dummy_pipe_fds[0])
|
|
|
|
close(pt->dummy_pipe_fds[0]);
|
|
|
|
if (pt->dummy_pipe_fds[1])
|
|
|
|
close(pt->dummy_pipe_fds[1]);
|
2016-01-19 03:34:24 +08:00
|
|
|
pt++;
|
|
|
|
}
|
context deprecation
1) This makes lwsws run a parent process with the original permissions.
But this process is only able to respond to SIGHUP, it doesn't do anything
else.
2) You can send this parent process a SIGHUP now to cause it to
- close listening sockets in existing lwsws processes
- mark those processes as to exit when the number of active connections
on the falls to zero
- spawn a fresh child process from scratch, using latest configuration
file content, latest plugins, etc. It can now reopen listening sockets
if it chooses to, or open different listen ports or whatever.
Notes:
1) lws_context_destroy() has been split into two pieces... the reason for
the split is the first part closes the per-vhost protocols, but since
they may have created libuv objects in the per-vhost protocol storage,
these cannot be freed until after the loop has been run.
That's the purpose of the second part of the context destruction,
lws_context_destroy2().
For compatibility, if you are not using libuv, the first part calls the
second part. However if you are using libuv, you must now call the
second part from your own main.c after the first part.
2016-12-16 07:37:43 +08:00
|
|
|
if (!context->fd_random)
|
|
|
|
lwsl_err("ZERO RANDOM FD\n");
|
|
|
|
if (context->fd_random != LWS_INVALID_FILE)
|
|
|
|
close(context->fd_random);
|
2014-04-02 14:25:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* cast a struct sockaddr_in6 * into addr for ipv6 */
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE int
|
ah owns rxbuf
This is intended to solve a longstanding problem with the
relationship between http/1.1 keep-alive and the service
loop.
Ah now contain an rx buffer which is used during header
processing, and the ah may not be detached from the wsi
until the rx buffer is exhausted.
Having the rx buffer in the ah means we can delay using the
rx until a later service loop.
Ah which have pending rx force POLLIN service on the wsi
they are attached to automatically, so we can interleave
general service / connections with draining each ah rx
buffer.
The possible http/1.1 situations and their dispositions are:
1) exactly one set of http headers come. After processing,
the ah is detached since no pending rx left. If more
headers come later, a fresh ah is aqcuired when available
and the rx flow control blocks the read until then.
2) more that one whole set of headers come and we remain in
http mode (no upgrade). The ah is left attached and
returns to the service loop after the first set of headers.
We will get forced service due to the ah having pending
content (respecting flowcontrol) and process the pending
rx in the ah. If we use it all up, we will detach the
ah.
3) one set of http headers come with ws traffic appended.
We service the headers, do the upgrade, and keep the ah
until the remaining ws content is used. When we
exhausted the ws traffix in the ah rx buffer, we
detach the ah.
Since there can be any amount of http/1.1 pipelining on a
connection, and each may be expensive to service, it's now
enforced there is a return to the service loop after each
header set is serviced on a connection.
When I added the forced service for ah with pending buffering,
I added support for it to the windows plat code. However this
is untested.
Signed-off-by: Andy Green <andy.green@linaro.org>
2016-02-15 12:37:04 +08:00
|
|
|
lws_interface_to_sa(int ipv6, const char *ifname, struct sockaddr_in *addr,
|
|
|
|
size_t addrlen)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
|
|
|
int rc = -1;
|
|
|
|
|
|
|
|
struct ifaddrs *ifr;
|
|
|
|
struct ifaddrs *ifc;
|
|
|
|
#ifdef LWS_USE_IPV6
|
|
|
|
struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *)addr;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
getifaddrs(&ifr);
|
|
|
|
for (ifc = ifr; ifc != NULL && rc; ifc = ifc->ifa_next) {
|
|
|
|
if (!ifc->ifa_addr)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
lwsl_info(" interface %s vs %s\n", ifc->ifa_name, ifname);
|
|
|
|
|
|
|
|
if (strcmp(ifc->ifa_name, ifname))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
switch (ifc->ifa_addr->sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
#ifdef LWS_USE_IPV6
|
2016-01-16 12:09:38 +08:00
|
|
|
if (ipv6) {
|
2014-04-02 14:25:10 +08:00
|
|
|
/* map IPv4 to IPv6 */
|
|
|
|
bzero((char *)&addr6->sin6_addr,
|
|
|
|
sizeof(struct in6_addr));
|
|
|
|
addr6->sin6_addr.s6_addr[10] = 0xff;
|
|
|
|
addr6->sin6_addr.s6_addr[11] = 0xff;
|
|
|
|
memcpy(&addr6->sin6_addr.s6_addr[12],
|
|
|
|
&((struct sockaddr_in *)ifc->ifa_addr)->sin_addr,
|
|
|
|
sizeof(struct in_addr));
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
memcpy(addr,
|
|
|
|
(struct sockaddr_in *)ifc->ifa_addr,
|
|
|
|
sizeof(struct sockaddr_in));
|
|
|
|
break;
|
|
|
|
#ifdef LWS_USE_IPV6
|
|
|
|
case AF_INET6:
|
|
|
|
memcpy(&addr6->sin6_addr,
|
|
|
|
&((struct sockaddr_in6 *)ifc->ifa_addr)->sin6_addr,
|
|
|
|
sizeof(struct in6_addr));
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
rc = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
freeifaddrs(ifr);
|
2014-07-29 15:36:06 +03:00
|
|
|
|
2014-04-27 12:32:15 +02:00
|
|
|
if (rc == -1) {
|
2016-10-02 02:21:03 +03:00
|
|
|
/* check if bind to IP address */
|
2014-04-27 12:32:15 +02:00
|
|
|
#ifdef LWS_USE_IPV6
|
|
|
|
if (inet_pton(AF_INET6, ifname, &addr6->sin6_addr) == 1)
|
|
|
|
rc = 0;
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
if (inet_pton(AF_INET, ifname, &addr->sin_addr) == 1)
|
|
|
|
rc = 0;
|
|
|
|
}
|
|
|
|
|
2014-04-02 14:25:10 +08:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE void
|
2016-01-19 03:34:24 +08:00
|
|
|
lws_plat_insert_socket_into_fds(struct lws_context *context, struct lws *wsi)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
2016-01-19 03:34:24 +08:00
|
|
|
struct lws_context_per_thread *pt = &context->pt[(int)wsi->tsi];
|
|
|
|
|
2015-12-16 18:19:08 +08:00
|
|
|
lws_libev_io(wsi, LWS_EV_START | LWS_EV_READ);
|
2016-02-14 09:27:41 +08:00
|
|
|
lws_libuv_io(wsi, LWS_EV_START | LWS_EV_READ);
|
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
pt->fds[pt->fds_count++].revents = 0;
|
2014-04-02 14:25:10 +08:00
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE void
|
2015-12-04 11:08:32 +08:00
|
|
|
lws_plat_delete_socket_from_fds(struct lws_context *context,
|
|
|
|
struct lws *wsi, int m)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
2016-01-26 20:56:56 +08:00
|
|
|
struct lws_context_per_thread *pt = &context->pt[(int)wsi->tsi];
|
2016-05-04 08:27:56 +08:00
|
|
|
|
|
|
|
lws_libev_io(wsi, LWS_EV_STOP | LWS_EV_READ | LWS_EV_WRITE);
|
|
|
|
lws_libuv_io(wsi, LWS_EV_STOP | LWS_EV_READ | LWS_EV_WRITE);
|
|
|
|
|
2016-01-26 20:56:56 +08:00
|
|
|
pt->fds_count--;
|
2014-04-02 14:25:10 +08:00
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE void
|
2015-12-04 11:08:32 +08:00
|
|
|
lws_plat_service_periodic(struct lws_context *context)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
|
|
|
/* if our parent went down, don't linger around */
|
|
|
|
if (context->started_with_parent &&
|
2015-12-17 17:03:59 +08:00
|
|
|
kill(context->started_with_parent, 0) < 0)
|
2014-04-02 14:25:10 +08:00
|
|
|
kill(getpid(), SIGTERM);
|
|
|
|
}
|
|
|
|
|
2014-04-02 21:02:54 +08:00
|
|
|
LWS_VISIBLE int
|
2015-12-04 11:08:32 +08:00
|
|
|
lws_plat_change_pollfd(struct lws_context *context,
|
|
|
|
struct lws *wsi, struct lws_pollfd *pfd)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
LWS_VISIBLE const char *
|
|
|
|
lws_plat_inet_ntop(int af, const void *src, char *dst, int cnt)
|
|
|
|
{
|
|
|
|
return inet_ntop(af, src, dst, cnt);
|
|
|
|
}
|
|
|
|
|
2017-03-01 14:28:56 +08:00
|
|
|
LWS_VISIBLE lws_fop_fd_t
|
|
|
|
_lws_plat_file_open(const struct lws_plat_file_ops *fops, const char *filename,
|
2017-03-03 12:38:10 +08:00
|
|
|
const char *vpath, lws_fop_flags_t *flags)
|
2014-04-02 14:25:10 +08:00
|
|
|
{
|
|
|
|
struct stat stat_buf;
|
2017-02-12 18:15:15 +08:00
|
|
|
int ret = open(filename, (*flags) & LWS_FOP_FLAGS_MASK, 0664);
|
2017-02-25 12:42:45 +08:00
|
|
|
lws_fop_fd_t fop_fd;
|
2014-04-02 14:25:10 +08:00
|
|
|
|
|
|
|
if (ret < 0)
|
2017-02-25 12:42:45 +08:00
|
|
|
return NULL;
|
2014-04-02 14:25:10 +08:00
|
|
|
|
2017-02-25 12:42:45 +08:00
|
|
|
if (fstat(ret, &stat_buf) < 0)
|
|
|
|
goto bail;
|
|
|
|
|
|
|
|
fop_fd = malloc(sizeof(*fop_fd));
|
|
|
|
if (!fop_fd)
|
|
|
|
goto bail;
|
|
|
|
|
|
|
|
fop_fd->fops = fops;
|
2017-03-03 12:38:10 +08:00
|
|
|
fop_fd->flags = *flags;
|
2017-02-25 12:42:45 +08:00
|
|
|
fop_fd->fd = ret;
|
|
|
|
fop_fd->filesystem_priv = NULL; /* we don't use it */
|
2017-03-03 12:38:10 +08:00
|
|
|
fop_fd->len = stat_buf.st_size;
|
|
|
|
fop_fd->pos = 0;
|
2017-02-25 12:42:45 +08:00
|
|
|
|
|
|
|
return fop_fd;
|
|
|
|
|
|
|
|
bail:
|
|
|
|
close(ret);
|
|
|
|
return NULL;
|
2014-04-02 23:03:23 +08:00
|
|
|
}
|
2014-04-03 07:16:40 +08:00
|
|
|
|
2017-03-01 14:28:56 +08:00
|
|
|
LWS_VISIBLE int
|
2017-03-03 12:38:10 +08:00
|
|
|
_lws_plat_file_close(lws_fop_fd_t *fop_fd)
|
2014-07-29 15:36:06 +03:00
|
|
|
{
|
2017-03-03 12:38:10 +08:00
|
|
|
int fd = (*fop_fd)->fd;
|
|
|
|
|
|
|
|
free(*fop_fd);
|
|
|
|
*fop_fd = NULL;
|
2017-02-25 12:42:45 +08:00
|
|
|
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
return close(fd);
|
|
|
|
}
|
|
|
|
|
2017-03-01 14:28:56 +08:00
|
|
|
LWS_VISIBLE lws_fileofs_t
|
2017-02-25 12:42:45 +08:00
|
|
|
_lws_plat_file_seek_cur(lws_fop_fd_t fop_fd, lws_fileofs_t offset)
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
{
|
2017-03-03 12:38:10 +08:00
|
|
|
lws_fileofs_t r;
|
|
|
|
|
|
|
|
if (offset > 0 && offset > fop_fd->len - fop_fd->pos)
|
|
|
|
offset = fop_fd->len - fop_fd->pos;
|
|
|
|
|
|
|
|
if ((lws_fileofs_t)fop_fd->pos + offset < 0)
|
|
|
|
offset = -fop_fd->pos;
|
|
|
|
|
|
|
|
r = lseek(fop_fd->fd, offset, SEEK_CUR);
|
|
|
|
|
|
|
|
if (r >= 0)
|
|
|
|
fop_fd->pos = r;
|
|
|
|
else
|
|
|
|
lwsl_err("error seeking from cur %ld, offset %ld\n",
|
2017-03-09 18:48:55 +08:00
|
|
|
(long)fop_fd->pos, (long)offset);
|
2017-03-03 12:38:10 +08:00
|
|
|
|
|
|
|
return r;
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
}
|
|
|
|
|
2017-03-01 14:28:56 +08:00
|
|
|
LWS_VISIBLE int
|
2017-02-25 12:42:45 +08:00
|
|
|
_lws_plat_file_read(lws_fop_fd_t fop_fd, lws_filepos_t *amount,
|
|
|
|
uint8_t *buf, lws_filepos_t len)
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
{
|
|
|
|
long n;
|
|
|
|
|
2017-02-25 12:42:45 +08:00
|
|
|
n = read((int)fop_fd->fd, buf, len);
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
if (n == -1) {
|
|
|
|
*amount = 0;
|
|
|
|
return -1;
|
|
|
|
}
|
2017-03-03 12:38:10 +08:00
|
|
|
fop_fd->pos += n;
|
|
|
|
lwsl_debug("%s: read %ld of req %ld, pos %ld, len %ld\n", __func__, n,
|
2017-03-09 18:48:55 +08:00
|
|
|
(long)len, (long)fop_fd->pos, (long)fop_fd->len);
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
*amount = n;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-03-01 14:28:56 +08:00
|
|
|
LWS_VISIBLE int
|
2017-02-25 12:42:45 +08:00
|
|
|
_lws_plat_file_write(lws_fop_fd_t fop_fd, lws_filepos_t *amount,
|
|
|
|
uint8_t *buf, lws_filepos_t len)
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
{
|
|
|
|
long n;
|
|
|
|
|
2017-02-25 12:42:45 +08:00
|
|
|
n = write((int)fop_fd->fd, buf, len);
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
if (n == -1) {
|
|
|
|
*amount = 0;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-03-03 12:38:10 +08:00
|
|
|
fop_fd->pos += n;
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
*amount = n;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-04-06 16:15:40 +08:00
|
|
|
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
LWS_VISIBLE int
|
|
|
|
lws_plat_init(struct lws_context *context,
|
|
|
|
struct lws_context_creation_info *info)
|
|
|
|
{
|
2016-01-19 03:34:24 +08:00
|
|
|
struct lws_context_per_thread *pt = &context->pt[0];
|
|
|
|
int n = context->count_threads, fd;
|
|
|
|
|
|
|
|
/* master context has the global fd lookup array */
|
|
|
|
context->lws_lookup = lws_zalloc(sizeof(struct lws *) *
|
|
|
|
context->max_fds);
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
if (context->lws_lookup == NULL) {
|
2016-01-19 03:34:24 +08:00
|
|
|
lwsl_err("OOM on lws_lookup array for %d connections\n",
|
|
|
|
context->max_fds);
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-02-04 13:09:00 +01:00
|
|
|
lwsl_notice(" mem: platform fd map: %5lu bytes\n",
|
|
|
|
(unsigned long)(sizeof(struct lws *) * context->max_fds));
|
2016-01-19 03:34:24 +08:00
|
|
|
fd = open(SYSTEM_RANDOM_FILEPATH, O_RDONLY);
|
2015-12-25 12:44:12 +08:00
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
context->fd_random = fd;
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
if (context->fd_random < 0) {
|
|
|
|
lwsl_err("Unable to open random device %s %d\n",
|
2016-01-19 03:34:24 +08:00
|
|
|
SYSTEM_RANDOM_FILEPATH, context->fd_random);
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-02-14 09:27:41 +08:00
|
|
|
if (!lws_libev_init_fd_table(context) &&
|
|
|
|
!lws_libuv_init_fd_table(context)) {
|
2015-12-23 11:55:06 +08:00
|
|
|
/* otherwise libev handled it instead */
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
|
2016-01-19 03:34:24 +08:00
|
|
|
while (n--) {
|
|
|
|
if (pipe(pt->dummy_pipe_fds)) {
|
|
|
|
lwsl_err("Unable to create pipe\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* use the read end of pipe as first item */
|
|
|
|
pt->fds[0].fd = pt->dummy_pipe_fds[0];
|
|
|
|
pt->fds[0].events = LWS_POLLIN;
|
|
|
|
pt->fds[0].revents = 0;
|
|
|
|
pt->fds_count = 1;
|
|
|
|
pt++;
|
2015-12-23 11:55:06 +08:00
|
|
|
}
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
}
|
|
|
|
|
2016-04-06 16:15:40 +08:00
|
|
|
#ifdef LWS_WITH_PLUGINS
|
2016-05-02 10:03:25 +08:00
|
|
|
if (info->plugin_dirs)
|
|
|
|
lws_plat_plugins_init(context, info->plugin_dirs);
|
2016-04-06 16:15:40 +08:00
|
|
|
#endif
|
|
|
|
|
lws_plat_fd implement platform default handlers
This is a rewrite of the patch from Soapyman here
https://github.com/warmcat/libwebsockets/pull/363
The main changes compared to Soapyman's original patch are
- There's no new stuff in the info struct user code does any overrides
it may want to do explicitly after lws_context_create returns
- User overrides for file ops can call through (subclass) to the original
platform implementation using lws_get_fops_plat()
- A typedef is provided for plat-specific fd type
- Public helpers are provided to allow user code to be platform-independent
about file access, using the lws platform file operations underneath:
static inline lws_filefd_type
lws_plat_file_open(struct lws_plat_file_ops *fops, const char *filename,
unsigned long *filelen, int flags)
static inline int
lws_plat_file_close(struct lws_plat_file_ops *fops, lws_filefd_type fd)
static inline unsigned long
lws_plat_file_seek_cur(struct lws_plat_file_ops *fops, lws_filefd_type fd,
long offset_from_cur_pos)
static inline int
lws_plat_file_read(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
static inline int
lws_plat_file_write(struct lws_plat_file_ops *fops, lws_filefd_type fd,
unsigned long *amount, unsigned char *buf, unsigned long len)
There's example documentation and implementation in the test server.
Signed-off-by: Andy Green <andy.green@linaro.org>
2015-12-10 07:58:58 +08:00
|
|
|
return 0;
|
2014-04-02 23:03:23 +08:00
|
|
|
}
|