NeoMutt  2024-03-23-147-g885fbc
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
raw.c File Reference

Low-level socket handling. More...

#include "config.h"
#include <errno.h>
#include <fcntl.h>
#include <netdb.h>
#include <netinet/in.h>
#include <signal.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <unistd.h>
#include "private.h"
#include "mutt/lib.h"
#include "config/lib.h"
#include "core/lib.h"
#include "gui/lib.h"
#include "connaccount.h"
#include "connection.h"
#include "globals.h"
#include "address/lib.h"
#include <stdbool.h>
+ Include dependency graph for raw.c:

Go to the source code of this file.

Functions

static int socket_connect (int fd, struct sockaddr *sa)
 Set up to connect to a socket fd.
 
int raw_socket_open (struct Connection *conn)
 Open a socket - Implements Connection::open() -.
 
int raw_socket_read (struct Connection *conn, char *buf, size_t count)
 Read data from a socket - Implements Connection::read() -.
 
int raw_socket_write (struct Connection *conn, const char *buf, size_t count)
 Write data to a socket - Implements Connection::write() -.
 
int raw_socket_poll (struct Connection *conn, time_t wait_secs)
 Check if any data is waiting on a socket - Implements Connection::poll() -.
 
int raw_socket_close (struct Connection *conn)
 Close a socket - Implements Connection::close() -.
 

Detailed Description

Low-level socket handling.

Authors
  • Damien Riegel
  • Richard Russon
  • Pietro Cerutti

This program 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 General Public License for more details.

You should have received a copy of the GNU General Public License along with this program. If not, see http://www.gnu.org/licenses/.

Definition in file raw.c.

Function Documentation

◆ socket_connect()

static int socket_connect ( int  fd,
struct sockaddr *  sa 
)
static

Set up to connect to a socket fd.

Parameters
fdFile descriptor to connect with
saAddress info
Return values
0Success
>0An errno, e.g. EPERM
-1Error

Definition at line 66 of file raw.c.

67{
68 int sa_size;
69 int save_errno;
70 sigset_t set;
71 struct sigaction oldalrm = { 0 };
72 struct sigaction act = { 0 };
73
74 if (sa->sa_family == AF_INET)
75 sa_size = sizeof(struct sockaddr_in);
76#ifdef HAVE_GETADDRINFO
77 else if (sa->sa_family == AF_INET6)
78 sa_size = sizeof(struct sockaddr_in6);
79#endif
80 else
81 {
82 mutt_debug(LL_DEBUG1, "Unknown address family!\n");
83 return -1;
84 }
85
86 /* Batch mode does not call mutt_signal_init(), so ensure the alarm
87 * interrupts the connect call */
88 const short c_socket_timeout = cs_subset_number(NeoMutt->sub, "socket_timeout");
89 if (c_socket_timeout > 0)
90 {
91 sigemptyset(&act.sa_mask);
92 act.sa_handler = mutt_sig_empty_handler;
93#ifdef SA_INTERRUPT
94 act.sa_flags = SA_INTERRUPT;
95#else
96 act.sa_flags = 0;
97#endif
98 sigaction(SIGALRM, &act, &oldalrm);
99 alarm(c_socket_timeout);
100 }
101
103
104 /* FreeBSD's connect() does not respect SA_RESTART, meaning
105 * a SIGWINCH will cause the connect to fail. */
106 sigemptyset(&set);
107 sigaddset(&set, SIGWINCH);
108 sigprocmask(SIG_BLOCK, &set, NULL);
109
110 save_errno = 0;
111
112 if (c_socket_timeout > 0)
113 {
114 const struct timeval tv = { c_socket_timeout, 0 };
115 if (setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv)) < 0)
116 {
117 mutt_debug(LL_DEBUG2, "Cannot set socket receive timeout. errno: %d\n", errno);
118 }
119 if (setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv)) < 0)
120 {
121 mutt_debug(LL_DEBUG2, "Cannot set socket send timeout. errno: %d\n", errno);
122 }
123 }
124
125 if (connect(fd, sa, sa_size) < 0)
126 {
127 save_errno = errno;
128 mutt_debug(LL_DEBUG2, "Connection failed. errno: %d\n", errno);
129 SigInt = false; /* reset in case we caught SIGINTR while in connect() */
130 }
131
132 if (c_socket_timeout > 0)
133 {
134 alarm(0);
135 sigaction(SIGALRM, &oldalrm, NULL);
136 }
138 sigprocmask(SIG_UNBLOCK, &set, NULL);
139
140 return save_errno;
141}
short cs_subset_number(const struct ConfigSubset *sub, const char *name)
Get a number config item by name.
Definition: helpers.c:144
#define mutt_debug(LEVEL,...)
Definition: logging2.h:89
@ LL_DEBUG2
Log at debug level 2.
Definition: logging2.h:44
@ LL_DEBUG1
Log at debug level 1.
Definition: logging2.h:43
void mutt_sig_empty_handler(int sig)
Dummy signal handler.
Definition: signal.c:73
volatile sig_atomic_t SigInt
true after SIGINT is received
Definition: signal.c:63
void mutt_sig_allow_interrupt(bool allow)
Allow/disallow Ctrl-C (SIGINT)
Definition: signal.c:254
Container for Accounts, Notifications.
Definition: neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition: neomutt.h:45
+ Here is the call graph for this function:
+ Here is the caller graph for this function: