/* ce-server.c - An Assuan testbed for W32CE; server code
Copyright (C) 2010 Free Software Foundation, Inc.
This file is part of Assuan.
Assuan is free software; you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as
published by the Free Software Foundation; either version 3 of
the License, or (at your option) any later version.
Assuan is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this program; if not, see .
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include
#include
#include
#include
#include
#include
#ifdef HAVE_W32_SYSTEM
# define WIN32_LEAN_AND_MEAN
# include
#else
# include
# include
# include
# include
# ifdef HAVE_SYS_SELECT_H
# include
# endif
#endif
#include
#ifdef HAVE_W32CE_SYSTEM
# ifndef FILE_ATTRIBUTE_ROMSTATICREF
# define FILE_ATTRIBUTE_ROMSTATICREF FILE_ATTRIBUTE_OFFLINE
# endif
extern BOOL GetStdioPathW (int, wchar_t *, DWORD *);
extern BOOL SetStdioPathW (int, const wchar_t *);
#endif /*!HAVE_W32CE_SYSTEM*/
#include "../src/assuan.h"
#include "common.h"
/* The port we are using by default. */
static short server_port = 15898;
/* Flag set to indicate a shutdown. */
static int shutdown_pending;
/* An object to keep track of file descriptors. */
struct fdinfo_s
{
struct fdinfo_s *next;
assuan_fd_t fd; /* The descriptor. */
};
typedef struct fdinfo_s *fdinfo_t;
/* The local state of a connection. */
struct state_s
{
/* The current working directory - access using get_cwd(). */
char *cwd;
/* If valid, a socket in listening state created by the dataport
command. */
assuan_fd_t dataport_listen_fd;
/* If valid the socket accepted for the dataport. */
assuan_fd_t dataport_accepted_fd;
/* The error code from a dataport accept operation. */
gpg_error_t dataport_accept_err;
/* A list of all unused descriptors created by dataport commands. */
fdinfo_t dataport_fds;
/* The descriptor set by the DATAPORT command. */
assuan_fd_t dataport_fd;
};
typedef struct state_s *state_t;
/* Local prototypes. */
static gpg_error_t cmd_newdataport_cont (void *opaque, gpg_error_t err,
unsigned char *data, size_t datalen);
/* A wrapper around read to make it work under Windows with HANDLES
and socket descriptors. Takes care of EINTR on POSIX. */
static int
my_read (assuan_fd_t fd, void *buffer, size_t size)
{
int res;
#ifdef HAVE_W32_SYSTEM
res = recv (HANDLE2SOCKET (fd), buffer, size, 0);
if (res == -1)
{
switch (WSAGetLastError ())
{
case WSAENOTSOCK:
{
DWORD nread = 0;
res = ReadFile (fd, buffer, size, &nread, NULL);
if (!res)
{
switch (GetLastError ())
{
case ERROR_BROKEN_PIPE:
gpg_err_set_errno (EPIPE);
break;
default:
gpg_err_set_errno (EIO);
}
res = -1;
}
else
res = (int) nread;
}
break;
case WSAEWOULDBLOCK:
gpg_err_set_errno (EAGAIN);
break;
case ERROR_BROKEN_PIPE:
gpg_err_set_errno (EPIPE);
break;
default:
gpg_err_set_errno (EIO);
break;
}
}
return res;
#else /*!HAVE_W32_SYSTEM*/
do
res = read (fd, buffer, size);
while (res == -1 && errno == EINTR);
return res;
#endif /*!HAVE_W32_SYSTEM*/
}
/* Extended version of write(2) to guarantee that all bytes are
written. Returns 0 on success or -1 and ERRNO on failure. NOTE:
This function does not return the number of bytes written, so any
error must be treated as fatal for this connection as the state of
the receiver is unknown. This works best if blocking is allowed
(so EAGAIN cannot occur). Under Windows this function handles
socket descriptors and system handles. */
static int
my_writen (assuan_fd_t fd, const char *buffer, size_t length)
{
while (length)
{
int nwritten;
#ifdef HAVE_W32_SYSTEM
nwritten = send (HANDLE2SOCKET (fd), buffer, length, 0);
if (nwritten == -1 && WSAGetLastError () == WSAENOTSOCK)
{
DWORD nwrite;
nwritten = WriteFile (fd, buffer, length, &nwrite, NULL);
if (!nwritten)
{
switch (GetLastError ())
{
case ERROR_BROKEN_PIPE:
case ERROR_NO_DATA:
gpg_err_set_errno (EPIPE);
break;
default:
gpg_err_set_errno (EIO);
break;
}
nwritten= -1;
}
else
nwritten = (int)nwrite;
}
#else /*!HAVE_W32_SYSTEM*/
nwritten = write (fd, buffer, length);
#endif /*!HAVE_W32_SYSTEM*/
if (nwritten < 0)
{
if (errno == EINTR)
continue;
return -1; /* write error */
}
length -= nwritten;
buffer += nwritten;
}
return 0; /* okay */
}
static state_t
new_state (void)
{
state_t state = xcalloc (1, sizeof *state);
state->dataport_listen_fd = ASSUAN_INVALID_FD;
state->dataport_accepted_fd = ASSUAN_INVALID_FD;
state->dataport_fd = ASSUAN_INVALID_FD;
return state;
}
static void
release_state (state_t state)
{
fdinfo_t fi, fi2;
if (!state)
return;
xfree (state->cwd);
if (state->dataport_fd != ASSUAN_INVALID_FD)
assuan_sock_close (state->dataport_fd);
if (state->dataport_listen_fd != ASSUAN_INVALID_FD)
assuan_sock_close (state->dataport_listen_fd);
if (state->dataport_accepted_fd != ASSUAN_INVALID_FD)
assuan_sock_close (state->dataport_accepted_fd);
for (fi=state->dataport_fds; fi; fi = fi2)
{
fi2 = fi->next;
if (fi->fd != ASSUAN_INVALID_FD)
assuan_sock_close (fi->fd);
}
xfree (state);
}
/* Helper to print a message while leaving a command and to
acknowledge the command. */
static gpg_error_t
leave_cmd (assuan_context_t ctx, gpg_error_t err)
{
if (err)
{
const char *name = assuan_get_command_name (ctx);
if (!name)
name = "?";
if (gpg_err_source (err) == GPG_ERR_SOURCE_DEFAULT)
log_error ("command '%s' failed: %s\n", name, gpg_strerror (err));
else
log_error ("command '%s' failed: %s <%s>\n", name,
gpg_strerror (err), gpg_strsource (err));
}
return assuan_process_done (ctx, err);
}
#ifdef HAVE_W32CE_SYSTEM
static char *
wchar_to_utf8 (const wchar_t *string)
{
int n;
size_t length = wcslen (string);
char *result;
n = WideCharToMultiByte (CP_UTF8, 0, string, length, NULL, 0, NULL, NULL);
if (n < 0 || (n+1) <= 0)
log_fatal ("WideCharToMultiByte failed\n");
result = xmalloc (n+1);
n = WideCharToMultiByte (CP_ACP, 0, string, length, result, n, NULL, NULL);
if (n < 0)
log_fatal ("WideCharToMultiByte failed\n");
result[n] = 0;
return result;
}
static wchar_t *
utf8_to_wchar (const char *string)
{
int n;
size_t length = strlen (string);
wchar_t *result;
size_t nbytes;
n = MultiByteToWideChar (CP_UTF8, 0, string, length, NULL, 0);
if (n < 0 || (n+1) <= 0)
log_fatal ("MultiByteToWideChar failed\n");
nbytes = (size_t)(n+1) * sizeof(*result);
if (nbytes / sizeof(*result) != (n+1))
log_fatal ("utf8_to_wchar: integer overflow\n");
result = xmalloc (nbytes);
n = MultiByteToWideChar (CP_UTF8, 0, string, length, result, n);
if (n < 0)
log_fatal ("MultiByteToWideChar failed\n");
result[n] = 0;
return result;
}
#endif /*HAVE_W32CE_SYSTEM*/
#ifndef HAVE_W32CE_SYSTEM
static char *
gnu_getcwd (void)
{
size_t size = 100;
while (1)
{
char *buffer = xmalloc (size);
if (getcwd (buffer, size) == buffer)
return buffer;
xfree (buffer);
if (errno != ERANGE)
return 0;
size *= 2;
}
}
#endif /*!HAVE_W32CE_SYSTEM*/
/* Return the current working directory. The returned string is valid
as long as STATE->cwd is not changed. */
static const char *
get_cwd (state_t state)
{
if (!state->cwd)
{
/* No working directory yet. On WindowsCE make it the module
directory of this process. */
#ifdef HAVE_W32_SYSTEM
char *p;
#endif
#ifdef HAVE_W32CE_SYSTEM
wchar_t buf[MAX_PATH+1];
size_t n;
n = GetModuleFileName (NULL, buf, MAX_PATH);
if (!n)
state->cwd = xstrdup ("/");
else
{
buf[n] = 0;
state->cwd = wchar_to_utf8 (buf);
p = strrchr (state->cwd, '\\');
if (p)
*p = 0;
}
#else
state->cwd = gnu_getcwd ();
#endif
#ifdef HAVE_W32_SYSTEM
for (p=state->cwd; *p; p++)
if (*p == '\\')
*p = '/';
#endif /*HAVE_W32_SYSTEM*/
}
return state->cwd;
}
static gpg_error_t
reset_notify (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
fdinfo_t fi, fi2;
/* Close all lingering dataport connections. */
for (fi=state->dataport_fds; fi; fi = fi2)
{
fi2 = fi->next;
if (fi->fd != ASSUAN_INVALID_FD)
assuan_sock_close (fi->fd);
}
state->dataport_fds = NULL;
return 0;
}
static gpg_error_t
input_notify (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
assuan_fd_t fd = assuan_get_input_fd (ctx);
fdinfo_t fi;
if (fd != ASSUAN_INVALID_FD)
{
/* The fd is now in use use - remove it from the unused list. */
for (fi=state->dataport_fds; fi; fi = fi->next)
if (fi->fd == fd)
fi->fd = ASSUAN_INVALID_FD;
}
return 0;
}
static gpg_error_t
output_notify (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
assuan_fd_t fd = assuan_get_output_fd (ctx);
fdinfo_t fi;
if (fd != ASSUAN_INVALID_FD)
{
/* The fd is now in use - remove it from the unused list. */
for (fi=state->dataport_fds; fi; fi = fi->next)
if (fi->fd == fd)
fi->fd = ASSUAN_INVALID_FD;
}
return 0;
}
static const char hlp_echo[] =
"ECHO \n"
"\n"
"Print LINE as data lines.\n";
static gpg_error_t
cmd_echo (assuan_context_t ctx, char *line)
{
gpg_error_t err;
err = assuan_send_data (ctx, line, strlen (line));
return leave_cmd (ctx, err);
}
static const char hlp_cat[] =
"CAT []\n"
"\n"
"Copy the content of FILENAME to the descriptor set by the OUTPUT\n"
"command. If no OUTPUT command has been given, send the content\n"
"using data lines. Without FILENAME take the content from the\n"
"descriptor set by the INPUT command; if a DATAPORT has been set\n"
"this descriptor is used for I/O and the INOPUT/OUTPUT descriptors\n"
"are not touched.";
static gpg_error_t
cmd_cat (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
gpg_error_t err = 0;
assuan_fd_t fd_in = ASSUAN_INVALID_FD;
assuan_fd_t fd_out = ASSUAN_INVALID_FD;
FILE *fp_in = NULL;
char buf[256];
size_t nread;
int use_dataport = 0;
if (*line)
{
fp_in = fopen (line, "rb");
if (!fp_in)
err = gpg_error_from_syserror ();
else
fd_out = assuan_get_output_fd (ctx);
}
else if (state->dataport_fd != ASSUAN_INVALID_FD)
{
use_dataport = 1;
fd_in = state->dataport_fd;
fd_out = state->dataport_fd;
}
else if ((fd_in = assuan_get_input_fd (ctx)) != ASSUAN_INVALID_FD)
{
/* This FD is actually a socket descriptor. We can't fdopen it
because under Windows we ust use recv(2) instead of read(2).
Note that on POSIX systems there is no difference between
libc file descriptors and socket descriptors. */
fd_out = assuan_get_output_fd (ctx);
}
else
err = gpg_error (GPG_ERR_ASS_NO_INPUT);
if (err)
goto leave;
do
{
if (fp_in)
{
nread = fread (buf, 1, sizeof buf, fp_in);
if (nread < sizeof buf)
{
if (ferror (fp_in))
err = gpg_error_from_syserror ();
else if (feof (fp_in))
err = gpg_error (GPG_ERR_EOF);
}
}
else
{
int n;
nread = 0;
n = my_read (fd_in, buf, sizeof buf);
if (n < 0)
err = gpg_error_from_syserror ();
else if (!n)
err = gpg_error (GPG_ERR_EOF);
else
nread = n;
}
if (fd_out != ASSUAN_INVALID_FD)
{
if (nread && my_writen (fd_out, buf, nread))
err = gpg_error_from_syserror ();
}
else if (nread)
err = assuan_send_data (ctx, buf, nread);
}
while (!err);
if (gpg_err_code (err) == GPG_ERR_EOF)
err = 0;
leave:
if (fp_in)
fclose (fp_in);
if (use_dataport)
{
if (state->dataport_fd != ASSUAN_INVALID_FD)
{
assuan_sock_close (state->dataport_fd);
state->dataport_fd = ASSUAN_INVALID_FD;
}
}
else
{
assuan_close_input_fd (ctx);
assuan_close_output_fd (ctx);
}
return leave_cmd (ctx, err);
}
static const char hlp_pwd[] =
"PWD\n"
"\n"
"Print the curent working directory of this session.\n";
static gpg_error_t
cmd_pwd (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
gpg_error_t err;
const char *string;
string = get_cwd (state);
err = assuan_send_data (ctx, string, strlen (string));
return leave_cmd (ctx, err);
}
static const char hlp_cd[] =
"CD [dir]\n"
"\n"
"Change the curretn directory of the session.\n";
static gpg_error_t
cmd_cd (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
gpg_error_t err = 0;
char *newdir, *p;
for (p=line; *p; p++)
if (*p == '\\')
*p = '/';
if (!*line)
{
xfree (state->cwd);
state->cwd = NULL;
get_cwd (state);
}
else
{
if (*line == '/')
newdir = xstrdup (line);
else
newdir = xstrconcat (get_cwd (state), "/", line, NULL);
while (strlen(newdir) > 1 && line[strlen(newdir)-1] == '/')
line[strlen(newdir)-1] = 0;
xfree (state->cwd);
state->cwd = newdir;
}
return leave_cmd (ctx, err);
}
#ifdef HAVE_W32CE_SYSTEM
static const char hlp_ls[] =
"LS []\n"
"\n"
"List the files described by PATTERN.\n";
static gpg_error_t
cmd_ls (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
gpg_error_t err;
WIN32_FIND_DATA fi;
char buf[500];
HANDLE hd;
char *p, *fname;
wchar_t *wfname;
if (!*line)
fname = xstrconcat (get_cwd (state), "/*", NULL);
else if (*line == '/' || *line == '\\')
fname = xstrdup (line);
else
fname = xstrconcat (get_cwd (state), "/", line, NULL);
for (p=fname; *p; p++)
if (*p == '/')
*p = '\\';
assuan_write_status (ctx, "PATTERN", fname);
wfname = utf8_to_wchar (fname);
xfree (fname);
hd = FindFirstFile (wfname, &fi);
free (wfname);
if (hd == INVALID_HANDLE_VALUE)
{
log_info ("FindFirstFile returned %d\n", GetLastError ());
err = gpg_error_from_syserror (); /* Works for W32CE. */
goto leave;
}
do
{
DWORD attr = fi.dwFileAttributes;
fname = wchar_to_utf8 (fi.cFileName);
snprintf (buf, sizeof buf,
"%c%c%c%c%c%c%c%c%c%c%c%c%c %7lu%c %s\n",
(attr & FILE_ATTRIBUTE_DIRECTORY)
? ((attr & FILE_ATTRIBUTE_DEVICE)? 'c':'d'):'-',
(attr & FILE_ATTRIBUTE_READONLY)? 'r':'-',
(attr & FILE_ATTRIBUTE_HIDDEN)? 'h':'-',
(attr & FILE_ATTRIBUTE_SYSTEM)? 's':'-',
(attr & FILE_ATTRIBUTE_ARCHIVE)? 'a':'-',
(attr & FILE_ATTRIBUTE_COMPRESSED)? 'c':'-',
(attr & FILE_ATTRIBUTE_ENCRYPTED)? 'e':'-',
(attr & FILE_ATTRIBUTE_INROM)? 'R':'-',
(attr & FILE_ATTRIBUTE_REPARSE_POINT)? 'P':'-',
(attr & FILE_ATTRIBUTE_ROMMODULE)? 'M':'-',
(attr & FILE_ATTRIBUTE_ROMSTATICREF)? 'R':'-',
(attr & FILE_ATTRIBUTE_SPARSE_FILE)? 'S':'-',
(attr & FILE_ATTRIBUTE_TEMPORARY)? 't':'-',
(unsigned long)fi.nFileSizeLow,
fi.nFileSizeHigh? 'X':' ',
fname);
free (fname);
err = assuan_send_data (ctx, buf, strlen (buf));
if (!err)
err = assuan_send_data (ctx, NULL, 0);
}
while (!err && FindNextFile (hd, &fi));
if (err)
;
else if (GetLastError () == ERROR_NO_MORE_FILES)
err = 0;
else
{
log_info ("FindNextFile returned %d\n", GetLastError ());
err = gpg_error_from_syserror ();
}
FindClose (hd);
leave:
return leave_cmd (ctx, err);
}
#endif /*HAVE_W32CE_SYSTEM*/
#ifdef HAVE_W32CE_SYSTEM
static const char hlp_run[] =
"RUN []\n"
"\n"
"Run the program in FILENAME with the arguments ARGS.\n"
"This creates a new process and waits for it to finish.\n"
"FIXME: The process' stdin is connected to the file set by the\n"
"INPUT command; stdout and stderr to the one set by OUTPUT.\n";
static gpg_error_t
cmd_run (assuan_context_t ctx, char *line)
{
/* state_t state = assuan_get_pointer (ctx); */
gpg_error_t err;
BOOL w32ret;
PROCESS_INFORMATION pi = { NULL, 0, 0, 0 };
char *p;
wchar_t *pgmname = NULL;
wchar_t *cmdline = NULL;
int code;
DWORD exc;
int idx;
struct {
HANDLE hd[2];
int oldname_valid;
wchar_t oldname[MAX_PATH];
} pipes[3];
for (idx=0; idx < 3; idx++)
{
pipes[idx].hd[0] = pipes[idx].hd[1] = INVALID_HANDLE_VALUE;
pipes[idx].oldname_valid = 0;
}
p = strchr (line, ' ');
if (p)
{
*p = 0;
pgmname = utf8_to_wchar (line);
for (p++; *p && *p == ' '; p++)
;
cmdline = utf8_to_wchar (p);
}
else
pgmname = utf8_to_wchar (line);
{
char *tmp1 = wchar_to_utf8 (pgmname);
char *tmp2 = wchar_to_utf8 (cmdline);
log_info ("CreateProcess, path=`%s' cmdline=`%s'\n", tmp1, tmp2);
xfree (tmp2);
xfree (tmp1);
}
/* Redirect the standard handles. */
/* Create pipes. */
for (idx=0; idx < 3; idx++)
{
if (!_assuan_w32ce_create_pipe (&pipes[idx].hd[0], &pipes[idx].hd[1],
NULL, 0))
{
err = gpg_error_from_syserror ();
log_error ("CreatePipe failed: %d\n", GetLastError ());
pipes[idx].hd[0] = pipes[idx].hd[1] = INVALID_HANDLE_VALUE;
goto leave;
}
}
/* Save currently assigned devices. */
for (idx=0; idx < 3; idx++)
{
DWORD dwlen = MAX_PATH;
if (!GetStdioPathW (idx, pipes[idx].oldname, &dwlen))
{
err = gpg_error_from_syserror ();
log_error ("GetStdioPath failed: %d\n", GetLastError ());
goto leave;
}
pipes[idx].oldname_valid = 1;
}
/* Connect the pipes. */
{
if (!SetStdioPathW (1, L"\\mystdout.log"))
{
err = gpg_error_from_syserror ();
log_error ("SetStdioPathW(%d) failed: %d\n", idx, GetLastError ());
goto leave;
}
if (!SetStdioPathW (2, L"\\mystderr.log"))
{
err = gpg_error_from_syserror ();
log_error ("SetStdioPathW(%d) failed: %d\n", idx, GetLastError ());
goto leave;
}
}
/* Create the process, restore the devices and check the error. */
w32ret = CreateProcess (pgmname, /* Program to start. */
cmdline, /* Command line arguments. */
NULL, /* Process security. Not used. */
NULL, /* Thread security. Not used. */
FALSE, /* Inherit handles. Not used. */
CREATE_SUSPENDED, /* Creation flags. */
NULL, /* Environment. Not used. */
NULL, /* Use current dir. Not used. */
NULL, /* Startup information. Not used. */
&pi /* Returns process information. */
);
for (idx=0; idx < 3; idx++)
{
if (pipes[idx].oldname_valid)
{
if (!SetStdioPathW (idx, pipes[idx].oldname))
log_error ("SetStdioPath(%d) failed during restore: %d\n",
idx, GetLastError ());
else
pipes[idx].oldname_valid = 0;
}
}
if (!w32ret)
{
/* Error checking after restore so that the messages are visible. */
log_error ("CreateProcess failed: %d\n", GetLastError ());
err = gpg_error_from_syserror ();
goto leave;
}
log_info ("CreateProcess ready: hProcess=%p hThread=%p"
" dwProcessID=%d dwThreadId=%d\n",
pi.hProcess, pi.hThread,
(int) pi.dwProcessId, (int) pi.dwThreadId);
ResumeThread (pi.hThread);
CloseHandle (pi.hThread);
code = WaitForSingleObject (pi.hProcess, INFINITE);
switch (code)
{
case WAIT_FAILED:
err = gpg_error_from_syserror ();;
log_error ("waiting for process %d to terminate failed: %d\n",
(int)pi.dwProcessId, GetLastError ());
break;
case WAIT_OBJECT_0:
if (!GetExitCodeProcess (pi.hProcess, &exc))
{
err = gpg_error_from_syserror ();;
log_error ("error getting exit code of process %d: %s\n",
(int)pi.dwProcessId, GetLastError () );
}
else if (exc)
{
log_info ("error running process: exit status %d\n", (int)exc);
err = gpg_error (GPG_ERR_GENERAL);
}
else
{
err = 0;
}
break;
default:
err = gpg_error_from_syserror ();;
log_error ("WaitForSingleObject returned unexpected "
"code %d for pid %d\n", code, (int)pi.dwProcessId);
break;
}
CloseHandle (pi.hProcess);
leave:
for (idx=0; idx < 3; idx++)
{
if (pipes[idx].oldname_valid)
{
if (!SetStdioPathW (idx, pipes[idx].oldname))
log_error ("SetStdioPath(%d) failed during restore: %d\n",
idx, GetLastError ());
else
pipes[idx].oldname_valid = 0;
}
}
for (idx=0; idx < 3; idx++)
{
if (pipes[idx].hd[0] != INVALID_HANDLE_VALUE)
CloseHandle (pipes[idx].hd[0]);
if (pipes[idx].hd[1] != INVALID_HANDLE_VALUE)
CloseHandle (pipes[idx].hd[1]);
}
xfree (cmdline);
xfree (pgmname);
return leave_cmd (ctx, err);
}
#endif /*HAVE_W32CE_SYSTEM*/
static const char hlp_newdataport[] =
"NEWDATAPORT\n"
"\n"
"Create a new dataport. The server creates a listening socket and\n"
"issues the inquiry:\n"
" INQUIRE CONNECT-TO \n"
"The client is expected to connect to PORT of the server and confirm\n"
"this by sending just an \"END\". In turn the server sends:\n"
" S FDINFO \n"
"With N being the local descriptor for the accepted connection. This\n"
"descriptor may now be used with INPUT or OUTPUT commands.";
struct cmd_dataport_locals
{
assuan_context_t ctx;
int passthru;
int port;
};
static gpg_error_t
cmd_newdataport (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
gpg_error_t err = 0;
struct sockaddr_in addr;
socklen_t addrlen;
struct cmd_dataport_locals *cont;
char inqline[100];
cont = xmalloc (sizeof *cont);
cont->ctx = ctx;
cont->passthru = 0;
cont->port = 0;
if (state->dataport_listen_fd != ASSUAN_INVALID_FD)
{
log_error ("Oops, still listening on a dataport socket\n");
state->dataport_listen_fd = ASSUAN_INVALID_FD;
}
if (state->dataport_accepted_fd != ASSUAN_INVALID_FD)
{
log_error ("Oops, still holding an accepted dataport socket\n");
state->dataport_accepted_fd = ASSUAN_INVALID_FD;
}
state->dataport_accept_err = 0;
state->dataport_listen_fd = assuan_sock_new (PF_INET, SOCK_STREAM, 0);
if (state->dataport_listen_fd == ASSUAN_INVALID_FD)
{
err = gpg_error_from_syserror ();
log_error ("socket() failed: %s\n", strerror (errno));
goto leave;
}
addr.sin_family = AF_INET;
addr.sin_port = 0;
addr.sin_addr.s_addr = htonl (INADDR_ANY);
if (assuan_sock_bind (state->dataport_listen_fd,
(struct sockaddr *)&addr, sizeof addr))
{
err = gpg_error_from_syserror ();
log_error ("listen() failed: %s\n", strerror (errno));
goto leave;
}
if (listen (HANDLE2SOCKET (state->dataport_listen_fd), 1))
{
err = gpg_error_from_syserror ();
log_error ("listen() failed: %s\n", strerror (errno));
goto leave;
}
addrlen = sizeof addr;
if (getsockname (HANDLE2SOCKET (state->dataport_listen_fd),
(struct sockaddr *)&addr, &addrlen))
{
err = gpg_error_from_syserror ();
log_error ("getsockname() failed: %s\n", strerror (errno));
goto leave;
}
cont->port = ntohs (addr.sin_port);
if (verbose)
log_info ("server now also listening on port %d\n", cont->port);
snprintf (inqline, sizeof inqline, "CONNECT-TO %d", cont->port);
err = assuan_inquire_ext (ctx, inqline, 0, cmd_newdataport_cont, cont);
if (!err)
return 0; /* Transfer to continuation. */
leave:
cont->passthru = 1;
return cmd_newdataport_cont (cont, err, NULL, 0);
}
/* Continuation used by cmd_newdataport. */
static gpg_error_t
cmd_newdataport_cont (void *opaque, gpg_error_t err,
unsigned char *data, size_t datalen)
{
struct cmd_dataport_locals *cont = opaque;
assuan_context_t ctx = cont->ctx;
state_t state = assuan_get_pointer (ctx);
char numbuf[35];
fdinfo_t fi;
if (cont->passthru || err)
goto leave;
err = state->dataport_accept_err;
if (err)
goto leave;
if (state->dataport_listen_fd != ASSUAN_INVALID_FD
|| state->dataport_accepted_fd == ASSUAN_INVALID_FD)
{
err = gpg_error (GPG_ERR_MISSING_ACTION);
goto leave;
}
for (fi = state->dataport_fds; fi; fi = fi->next)
if (fi->fd == ASSUAN_INVALID_FD)
break;
if (!fi)
{
fi = xcalloc (1, sizeof *fi);
fi->next = state->dataport_fds;
state->dataport_fds = fi;
}
fi->fd = state->dataport_accepted_fd;
state->dataport_accepted_fd = ASSUAN_INVALID_FD;
/* Note that under Windows the FD is the socket descriptor. Socket
descriptors are neither handles nor libc file descriptors. */
snprintf (numbuf, sizeof numbuf, "%d", HANDLE2SOCKET (fi->fd));
err = assuan_write_status (ctx, "FDINFO", numbuf);
leave:
if (state->dataport_listen_fd != ASSUAN_INVALID_FD)
{
assuan_sock_close (state->dataport_listen_fd);
state->dataport_listen_fd = ASSUAN_INVALID_FD;
}
if (state->dataport_accepted_fd != ASSUAN_INVALID_FD)
{
assuan_sock_close (state->dataport_accepted_fd);
state->dataport_accepted_fd = ASSUAN_INVALID_FD;
}
xfree (cont);
return leave_cmd (ctx, err);
}
static const char hlp_dataport[] =
"DATAPORT FD[=]\n"
"\n"
"Set the file descriptor to read and write data via port.\n"
"This is similar to the \"INPUT\" and \"OUTPUT\" commands\n"
"but useful for socketpairs.";
static gpg_error_t
cmd_dataport (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
gpg_error_t err;
assuan_fd_t fd;
if (state->dataport_fd != ASSUAN_INVALID_FD)
{
assuan_sock_close (state->dataport_fd);
state->dataport_fd = ASSUAN_INVALID_FD;
}
err = assuan_command_parse_fd (ctx, line, &fd);
if (!err && fd != ASSUAN_INVALID_FD)
{
fdinfo_t fi;
state->dataport_fd = fd;
/* The fd is now in use use - remove it from the unused list. */
for (fi=state->dataport_fds; fi; fi = fi->next)
if (fi->fd == fd)
fi->fd = ASSUAN_INVALID_FD;
}
return leave_cmd (ctx, err);
}
static const char hlp_getinfo[] =
"GETINFO \n"
"\n"
"Multipurpose function to return a variety of information.\n"
"Supported values for WHAT are:\n"
"\n"
" version - Return the version of the program.\n"
" pid - Return the process id of the server.\n"
" dataports - Return a list of usused dataports.";
static gpg_error_t
cmd_getinfo (assuan_context_t ctx, char *line)
{
state_t state = assuan_get_pointer (ctx);
gpg_error_t err = 0;
char numbuf[50];
if (!strcmp (line, "version"))
{
const char *s = VERSION;
err = assuan_send_data (ctx, s, strlen (s));
}
else if (!strcmp (line, "pid"))
{
snprintf (numbuf, sizeof numbuf, "%lu", (unsigned long)getpid ());
err = assuan_send_data (ctx, numbuf, strlen (numbuf));
}
else if (!strcmp (line, "dataports"))
{
fdinfo_t fi;
int any = 0;
for (fi=state->dataport_fds; !err && fi; fi = fi->next)
{
if (fi->fd != ASSUAN_INVALID_FD)
{
snprintf (numbuf, sizeof numbuf, "%s%d",
any? " ":"", HANDLE2SOCKET (fi->fd));
any = 1;
err = assuan_send_data (ctx, numbuf, strlen (numbuf));
}
}
}
else
err = gpg_error (GPG_ERR_ASS_PARAMETER);
return leave_cmd (ctx, err);
}
static const char hlp_shutdown[] =
"SHUTDOWN\n"
"\n"
"Shutdown the server process\n";
static gpg_error_t
cmd_shutdown (assuan_context_t ctx, char *line)
{
(void)line;
shutdown_pending = 1;
return leave_cmd (ctx, 0);;
}
static gpg_error_t
register_commands (assuan_context_t ctx)
{
static struct
{
const char *name;
gpg_error_t (*handler) (assuan_context_t, char *line);
const char * const help;
} table[] =
{
#ifdef HAVE_W32CE_SYSTEM
{ "LS", cmd_ls, hlp_ls },
{ "RUN", cmd_run, hlp_run },
#endif
{ "PWD", cmd_pwd, hlp_pwd },
{ "CD", cmd_cd, hlp_cd },
{ "ECHO", cmd_echo, hlp_echo },
{ "CAT", cmd_cat, hlp_cat },
{ "NEWDATAPORT", cmd_newdataport, hlp_newdataport },
{ "DATAPORT", cmd_dataport, hlp_dataport },
{ "INPUT", NULL },
{ "OUTPUT", NULL },
{ "GETINFO", cmd_getinfo, hlp_getinfo },
{ "SHUTDOWN", cmd_shutdown, hlp_shutdown },
{ NULL, NULL }
};
int i;
gpg_error_t rc;
for (i=0; table[i].name; i++)
{
rc = assuan_register_command (ctx, table[i].name,
table[i].handler, table[i].help);
if (rc)
return rc;
}
return 0;
}
static assuan_fd_t
get_connection_fd (assuan_context_t ctx)
{
assuan_fd_t fds[5];
if (assuan_get_active_fds (ctx, 0, fds, DIM (fds)) < 1)
log_fatal ("assuan_get_active_fds failed\n");
if (fds[0] == ASSUAN_INVALID_FD)
log_fatal ("assuan_get_active_fds returned invalid conenction fd\n");
return fds[0];
}
/* Startup the server. */
static void
server (void)
{
gpg_error_t err;
assuan_fd_t server_fd;
assuan_sock_nonce_t server_nonce;
int one = 1;
struct sockaddr_in name;
assuan_context_t ctx;
state_t state = NULL;
err = assuan_new (&ctx);
if (err)
log_fatal ("assuan_new failed: %s\n", gpg_strerror (err));
server_fd = assuan_sock_new (PF_INET, SOCK_STREAM, 0);
if (server_fd == ASSUAN_INVALID_FD)
log_fatal ("socket() failed: %s\n", strerror (errno));
if (setsockopt (HANDLE2SOCKET (server_fd),
SOL_SOCKET, SO_REUSEADDR, (void*)&one, sizeof one))
log_error ("setsockopt(SO_REUSEADDR) failed: %s\n", strerror (errno));
name.sin_family = AF_INET;
name.sin_port = htons (server_port);
name.sin_addr.s_addr = htonl (INADDR_ANY);
if (assuan_sock_bind (server_fd, (struct sockaddr *) &name, sizeof name))
log_fatal ("bind() failed: %s\n", strerror (errno));
if (assuan_sock_get_nonce ((struct sockaddr*)&name, sizeof name,
&server_nonce))
log_fatal ("assuan_sock_get_nonce failed: %s\n", strerror (errno));
/* Register the nonce with the context so that assuan_accept knows
about it. We can't do that directly in assuan_sock_bind because
we want these socket wrappers to be context neutral and drop in
replacement for the standard socket functions. */
assuan_set_sock_nonce (ctx, &server_nonce);
if (listen (HANDLE2SOCKET (server_fd), 5))
log_fatal ("listen() failed: %s\n", strerror (errno));
log_info ("server listening on port %hd\n", server_port);
err = assuan_init_socket_server (ctx, server_fd, 0);
if (err)
log_fatal ("assuan_init_socket_server failed: %s\n", gpg_strerror (err));
err = register_commands (ctx);
if (err)
log_fatal ("register_commands failed: %s\n", gpg_strerror(err));
if (debug)
assuan_set_log_stream (ctx, stderr);
assuan_register_reset_notify (ctx, reset_notify);
assuan_register_input_notify (ctx, input_notify);
assuan_register_output_notify (ctx, output_notify);
state = new_state ();
assuan_set_pointer (ctx, state);
while (!shutdown_pending)
{
int done;
err = assuan_accept (ctx);
if (err)
{
if (gpg_err_code (err) == GPG_ERR_EOF || err == -1)
log_error ("assuan_accept failed: %s\n", gpg_strerror (err));
break;
}
log_info ("client connected. Client's pid is %ld\n",
(long)assuan_get_pid (ctx));
do
{
/* We need to use select here so that we can accept
supplemental connections from the client as requested by
the DATAPORT command. */
fd_set rfds;
int connfd, datafd, max_fd;
connfd = HANDLE2SOCKET (get_connection_fd (ctx));
FD_ZERO (&rfds);
FD_SET (connfd, &rfds);
max_fd = connfd;
if (state->dataport_listen_fd != ASSUAN_INVALID_FD)
{
datafd = HANDLE2SOCKET (state->dataport_listen_fd);
FD_SET (datafd, &rfds);
if (datafd > max_fd)
max_fd = datafd;
}
else
datafd = -1;
if (select (max_fd + 1, &rfds, NULL, NULL, NULL) > 0)
{
if (datafd != -1 && FD_ISSET (datafd, &rfds))
{
struct sockaddr_in clnt_addr;
socklen_t len = sizeof clnt_addr;
int fd;
fd = accept (datafd, (struct sockaddr*)&clnt_addr, &len);
if (fd == -1)
{
err = gpg_err_code_from_syserror ();
assuan_sock_close (state->dataport_listen_fd);
state->dataport_listen_fd = ASSUAN_INVALID_FD;
log_error ("accepting on dataport failed: %s\n",
gpg_strerror (err));
state->dataport_accept_err = err;
err = 0;
}
else
{
/* No more need for the listening socket. */
assuan_sock_close (state->dataport_listen_fd);
state->dataport_listen_fd = ASSUAN_INVALID_FD;
/* Record the accepted fd. */
state->dataport_accept_err = 0;
state->dataport_accepted_fd = SOCKET2HANDLE (fd);
}
}
if (FD_ISSET (connfd, &rfds))
{
err = assuan_process_next (ctx, &done);
}
}
}
while (!err && !done && !shutdown_pending);
if (err)
log_error ("assuan_process failed: %s\n", gpg_strerror (err));
}
assuan_sock_close (server_fd);
assuan_release (ctx);
release_state (state);
}
/*
M A I N
*/
int
main (int argc, char **argv)
{
gpg_error_t err;
int last_argc = -1;
if (argc)
{
log_set_prefix (*argv);
argc--; argv++;
}
while (argc && last_argc != argc )
{
last_argc = argc;
if (!strcmp (*argv, "--help"))
{
printf (
"usage: %s [options]\n"
"\n"
"Options:\n"
" --verbose Show what is going on\n",
log_get_prefix ());
exit (0);
}
if (!strcmp (*argv, "--verbose"))
{
verbose = 1;
argc--; argv++;
}
else if (!strcmp (*argv, "--debug"))
{
verbose = debug = 1;
argc--; argv++;
}
}
assuan_set_assuan_log_prefix (log_prefix);
if (debug)
assuan_set_assuan_log_stream (stderr);
err = assuan_sock_init ();
if (err)
log_fatal ("assuan_sock_init failed: %s\n", gpg_strerror (err));
log_info ("server starting...\n");
server ();
log_info ("server finished\n");
assuan_sock_deinit ();
return errorcount ? 1 : 0;
}