3 * This file is part of LyX, the document processor.
4 * Licence details can be found in the file COPYING.
6 * \author João Luis M. Assirati
8 * Full author contact details are available in file CREDITS.
13 #include "support/socktools.h"
14 #include "support/FileName.h"
16 #if !defined (HAVE_FCNTL)
17 // We provide stubs because we don't (yet?) support the native OS API.
23 int listen(FileName const &, int)
34 } // namespace socktools
35 } // namespace support
38 #else // defined (HAVE_FCNTL)
40 #include "support/debug.h"
42 #ifdef HAVE_SYS_TYPES_H
43 # include <sys/types.h>
45 #include <sys/socket.h>
54 //needed for Mac OSX 10.5.2 Leopard
60 // This MACRO eppears to be defined only on Linux.
63 (sizeof (*(su)) - sizeof ((su)->sun_path) + strlen((su)->sun_path))
71 // Returns a local socket already in the "listen" state (or -1 in case
72 // of error). The first argument is the socket address, the second
73 // is the length of the queue for connections. If successful, a socket
74 // special file 'name' will be created in the filesystem.
75 int listen(FileName const & name, int queue)
77 int fd; // File descriptor for the socket
78 sockaddr_un addr; // Structure that hold the socket address
80 // We use 'localname' to fill 'addr'
81 string const localname = name.toFilesystemEncoding();
82 string::size_type len = localname.size();
83 // the field sun_path in sockaddr_un is a char[108]
85 LYXERR0("lyx: Socket address '" << name.absFileName() << "' too long.");
88 // Synonims for AF_UNIX are AF_LOCAL and AF_FILE
89 addr.sun_family = AF_UNIX;
90 localname.copy(addr.sun_path, 107);
91 addr.sun_path[len] = '\0';
93 // This creates a file descriptor for the socket
94 // Synonims for PF_UNIX are PF_LOCAL and PF_FILE
95 // For local sockets, the protocol is always 0
96 // socket() returns -1 in case of error
97 if ((fd = ::socket(PF_UNIX, SOCK_STREAM, 0))== -1) {
98 LYXERR0("lyx: Could not create socket descriptor: "
103 // Set NONBLOCK mode for the file descriptor
104 if (::fcntl(fd, F_SETFL, O_NONBLOCK) == -1) {
105 LYXERR0("lyx: Could not set NONBLOCK mode for socket descriptor: "
111 // bind() gives the local address 'name' for 'fd', also creating
112 // the socket special file in the filesystem. bind() returns -1
114 if ((::bind (fd, reinterpret_cast<sockaddr *>(&addr), SUN_LEN(&addr))) == -1) {
115 LYXERR0("lyx: Could not bind address '" << name.absFileName()
116 << "' to socket descriptor: " << strerror(errno));
122 // Puts the socket in listen state, that is, ready to accept
123 // connections. The second parameter of listen() defines the
124 // maximum length the queue of pending connections may grow to.
125 // It is not a restriction on the number of connections the socket
126 // can accept. Returns -1 in case of error
127 if (::listen (fd, queue) == -1) {
128 LYXERR0("lyx: Could not put socket in 'listen' state: "
138 // Returns a file descriptor for a new connection from the socket
139 // descriptor 'sd' (or -1 in case of error)
144 // Returns the new file descriptor or -1 in case of error
145 // Using null pointers for the second and third arguments
146 // dismiss all information about the connecting client
147 if ((fd = accept(sd, reinterpret_cast<sockaddr *>(0), reinterpret_cast<socklen_t *>(0))) == -1) {
148 LYXERR0("lyx: Could not accept connection: "
153 // Sets NONBLOCK mode for the file descriptor
154 if (::fcntl(fd, F_SETFL, O_NONBLOCK) == -1) {
155 LYXERR0("lyx: Could not set NONBLOCK mode for connection: "
163 } // namespace socktools
164 } // namespace support
167 #endif // defined (HAVE_FCNTL)