UnixFaxServer.c++
上传用户:weiyuanprp
上传日期:2020-05-20
资源大小:1169k
文件大小:6k
源码类别:

传真(Fax)编程

开发平台:

C/C++

  1. /* $Id: UnixFaxServer.c++,v 1.1.1.1 2005/11/11 21:32:03 faxguy Exp $ */
  2. /*
  3.  * Copyright (c) 1995-1996 Sam Leffler
  4.  * Copyright (c) 1995-1996 Silicon Graphics, Inc.
  5.  * HylaFAX is a trademark of Silicon Graphics
  6.  *
  7.  * Permission to use, copy, modify, distribute, and sell this software and 
  8.  * its documentation for any purpose is hereby granted without fee, provided
  9.  * that (i) the above copyright notices and this permission notice appear in
  10.  * all copies of the software and related documentation, and (ii) the names of
  11.  * Sam Leffler and Silicon Graphics may not be used in any advertising or
  12.  * publicity relating to the software without the specific, prior written
  13.  * permission of Sam Leffler and Silicon Graphics.
  14.  * 
  15.  * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND, 
  16.  * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY 
  17.  * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  
  18.  * 
  19.  * IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
  20.  * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
  21.  * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
  22.  * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF 
  23.  * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE 
  24.  * OF THIS SOFTWARE.
  25.  */
  26. /*
  27.  * XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
  28.  *
  29.  * This stuff does not work; it is here as a placeholder in
  30.  * case it becomes worthwhile to add UNIX domain socket support.
  31.  *
  32.  * XXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
  33.  */
  34. #if CONFIG_UNIXTRANSPORT
  35. #include "Dispatcher.h"
  36. #include "UnixFaxServer.h"
  37. #include "Sys.h"
  38. #include "Socket.h"
  39. #include "config.h"
  40. UnixSuperServer::UnixSuperServer(const char* f, int bl)
  41.     : SuperServer("UNIX", bl)
  42.     , fileName(f)
  43. {}
  44. UnixSuperServer::~UnixSuperServer()
  45. {
  46.     if (fileName != "")
  47. Sys::unlink(fileName);
  48. }
  49. bool
  50. UnixSuperServer::startServer(void)
  51. {
  52.     int s = socket(AF_UNIX, SOCK_STREAM, 0);
  53.     if (s >= 0) {
  54. struct sockaddr_un Sun;
  55. Sun.sun_family = AF_UNIX;
  56. strncpy(Sun.sun_path, fileName, sizeof (Sun.sun_path));
  57. if (Socket::bind(s, &Sun, sizeof (Sun)) >= 0) {
  58. #ifdef HAS_FCHMOD
  59.     (void) fchmod(s, 0622);
  60. #else
  61.     (void) Sys::chmod(fileName, 0622);
  62. #endif
  63.     (void) listen(s, getBacklog());
  64.     Dispatcher::instance().link(s, Dispatcher::ReadMask, this);
  65.     return (true);
  66. }
  67. Sys::close(s);
  68. logError("%s HylaFAX: bind (port %s): %m",
  69.     getKind(), (const char*) fileName);
  70. fileName = ""; // don't try to unlink file
  71.     } else
  72. logError("%s HylaFAX: socket: %m", getKind());
  73.     return (false);
  74. }
  75. HylaFAXServer* UnixSuperServer::newChild(void) { return new UnixFaxServer; }
  76. UnixFaxServer::UnixFaxServer()
  77. {
  78.     usedefault = true;
  79. }
  80. UnixFaxServer::~UnixFaxServer() {}
  81. void
  82. UnixFaxServer::initServer(void)
  83. {
  84.     HylaFAXServer::initServer();
  85.     usedefault = true;
  86. }
  87. void
  88. UnixFaxServer::open(void)
  89. {
  90.     remotehost = hostname; // always on same machine
  91.     remoteaddr = "local-UNIX"; // XXX
  92.     Dispatcher::instance().link(STDIN_FILENO, Dispatcher::ReadMask, this);
  93.     HylaFAXServer::open();
  94. }
  95. void
  96. UnixFaxServer::passiveCmd(void)
  97. {
  98.     perror_reply(425, "Cannot open passive connection "
  99. "(makes no sense with UNIX domain sockets)", errno);
  100. }
  101. void
  102. UnixFaxServer::netStatus(FILE* fd)
  103. {
  104.     if (data != -1)
  105.         fprintf(fd, "    Client data connection openrn");
  106.     else
  107.         fprintf(fd, "    No client data connectionrn");
  108. }
  109. /*
  110.  * Creat a socket for a data transfer.
  111.  */
  112. FILE*
  113. UnixFaxServer::getDataSocket(const char* mode)
  114. {
  115.     if (data >= 0)
  116.         return (fdopen(data, mode));
  117.     int s = socket(AF_UNIX, SOCK_STREAM, 0);
  118.     if (s >= 0) {
  119. /* anchor socket to avoid multi-homing problems */
  120. data_source.sun_family = AF_UNIX;
  121. strcpy(data_source.sun_path, ctrl_addr.sun_path);
  122. if (bind(s, (struct sockaddr*) &data_source, sizeof (data_source)) >= 0) {
  123.     return (fdopen(s, mode));
  124. }
  125.     }
  126.     (void) Sys::close(s);
  127.     return (NULL);
  128. }
  129. bool
  130. UnixFaxServer::dataConnect(void)
  131. {
  132.     return Socket::connect(data, &data_dest,sizeof (data_dest)) >= 0;
  133. }
  134. /*
  135.  * Establish a data connection for a file transfer operation.
  136.  */
  137. FILE*
  138. UnixFaxServer::openDataConn(const char* mode, int& code)
  139. {
  140.     byte_count = 0;
  141.     if (pdata >= 0) {
  142.         struct sockaddr_un from;
  143.         int fromlen = sizeof(from);
  144.         int s = accept(pdata, (struct sockaddr*) &from, &fromlen);
  145.         if (s < 0) {
  146.             reply(425, "Cannot open data connection.");
  147.             (void) Sys::close(pdata);
  148.             pdata = -1;
  149.             return (NULL);
  150.         }
  151.         (void) Sys::close(pdata);
  152.         pdata = s;
  153. code = 150;
  154.         return (fdopen(pdata, mode));
  155.     }
  156.     if (data >= 0) {
  157. code = 125;
  158.         usedefault = 1;
  159.         return (fdopen(data, mode));
  160.     }
  161.     if (usedefault)
  162.         data_dest = peer_addr;
  163.     usedefault = 1;
  164.     FILE* file = getDataSocket(mode);
  165.     if (file == NULL) {
  166.         reply(425, "Cannot create data socket (%s): %s.",
  167.       data_source.sun_path, strerror(errno));
  168. return (NULL);
  169.     }
  170.     data = fileno(file);
  171.     if (!dataConnect()) {
  172. perror_reply(425, "Cannot build data connection", errno);
  173. fclose(file);
  174. data = -1;
  175. return (NULL);
  176.     }
  177.     code = 150;
  178.     return (file);
  179. }
  180. bool
  181. UnixFaxServer::hostPort()
  182. {
  183.     fxStr s;
  184.     if (pathname(s)) {
  185. data_dest.sun_family = AF_UNIX;
  186. strncpy(data_dest.sun_path, s, sizeof (data_dest.sun_path));
  187. return (true);
  188.     } else
  189. return (false);
  190. }
  191. void
  192. UnixFaxServer::portCmd(void)
  193. {
  194.     logcmd(T_PORT, "%s", data_dest.sun_path);
  195.     usedefault = false;
  196.     if (pdata >= 0)
  197. (void) Sys::close(pdata), pdata = -1;
  198.     reply(200, "PORT command successful.");
  199. }
  200. #endif