| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /* GIO - GLib Input, Output and Streaming Library
 | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Copyright (C) 2008 Christian Kellner, Samuel Cormier-Iijima | 
					
						
							|  |  |  |  |  * Copyright © 2009 Codethink Limited | 
					
						
							|  |  |  |  |  * Copyright © 2009 Red Hat, Inc | 
					
						
							| 
									
										
										
										
											2015-06-12 08:47:37 +01:00
										 |  |  |  |  * Copyright © 2015 Collabora, Ltd. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * This library 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 2 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This library 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 | 
					
						
							| 
									
										
										
										
											2014-01-23 12:58:29 +01:00
										 |  |  |  |  * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Authors: Christian Kellner <gicmo@gnome.org> | 
					
						
							|  |  |  |  |  *          Samuel Cormier-Iijima <sciyoshi@gmail.com> | 
					
						
							|  |  |  |  |  *          Ryan Lortie <desrt@desrt.ca> | 
					
						
							|  |  |  |  |  *          Alexander Larsson <alexl@redhat.com> | 
					
						
							| 
									
										
										
										
											2015-06-12 08:47:37 +01:00
										 |  |  |  |  *          Philip Withnall <philip.withnall@collabora.co.uk> | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "config.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-05 17:32:12 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "gsocket.h"
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-03 09:52:10 -04:00
										 |  |  |  | #ifdef G_OS_UNIX
 | 
					
						
							|  |  |  |  | #include "glib-unix.h"
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2009-05-17 20:44:54 -04:00
										 |  |  |  | #include <signal.h>
 | 
					
						
							| 
									
										
										
										
											2009-05-15 09:10:23 +02:00
										 |  |  |  | #include <string.h>
 | 
					
						
							|  |  |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  | # include <fcntl.h>
 | 
					
						
							|  |  |  |  | # include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  | # include <sys/ioctl.h>
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-06 11:55:59 -04:00
										 |  |  |  | #ifdef HAVE_SYS_FILIO_H
 | 
					
						
							|  |  |  |  | # include <sys/filio.h>
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
											
												Require POSIX.1 (1990) compliance on unix
Assume unix platforms support the original POSIX.1 standard.
Specifically, assume that if G_OS_UNIX, then we have chown(),
getcwd(), getgrgid(), getpwuid(), link(), <grp.h>, <pwd.h>,
<sys/types.h>, <sys/uio.h>, <sys/wait.h>, and <unistd.h>.
Additionally, since all versions of Windows that we care about also
have <sys/types.h>, we can remove HAVE_SYS_TYPES_H checks everywhere.
Also remove one include of <sys/times.h>, and the corresponding
configure check, since the include is not currently needed (and may
always have just been a typo for <sys/time.h>).
https://bugzilla.gnome.org/show_bug.cgi?id=710519
											
										 
											2013-10-19 13:03:59 -04:00
										 |  |  |  | #ifdef G_OS_UNIX
 | 
					
						
							| 
									
										
										
										
											2009-06-14 23:55:12 -04:00
										 |  |  |  | #include <sys/uio.h>
 | 
					
						
							| 
									
										
										
										
											2009-06-19 23:18:09 -04:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-06-14 23:55:12 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #include "gcancellable.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:50:42 +01:00
										 |  |  |  | #include "gdatagrambased.h"
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #include "gioenumtypes.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | #include "ginetaddress.h"
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #include "ginitable.h"
 | 
					
						
							|  |  |  |  | #include "gioerror.h"
 | 
					
						
							|  |  |  |  | #include "gioenums.h"
 | 
					
						
							|  |  |  |  | #include "gioerror.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-22 16:39:49 -05:00
										 |  |  |  | #include "gnetworkingprivate.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | #include "gsocketaddress.h"
 | 
					
						
							|  |  |  |  | #include "gsocketcontrolmessage.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  | #include "gcredentials.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-18 13:40:09 -04:00
										 |  |  |  | #include "gcredentialsprivate.h"
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #include "glibintl.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-05 16:13:03 +08:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  | /* For Windows XP runtime compatibility, but use the system's if_nametoindex() if available */ | 
					
						
							|  |  |  |  | #include "gwin32networking.h"
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * SECTION:gsocket | 
					
						
							| 
									
										
										
										
											2009-05-18 13:02:11 +02:00
										 |  |  |  |  * @short_description: Low-level socket object | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @include: gio/gio.h | 
					
						
							| 
									
										
										
										
											2014-02-09 02:07:26 -05:00
										 |  |  |  |  * @see_also: #GInitable, [<gnetworking.h>][gio-gnetworking.h] | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * A #GSocket is a low-level networking primitive. It is a more or less | 
					
						
							|  |  |  |  |  * direct mapping of the BSD socket API in a portable GObject based API. | 
					
						
							| 
									
										
										
										
											2009-05-28 14:40:04 -04:00
										 |  |  |  |  * It supports both the UNIX socket implementations and winsock2 on Windows. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * #GSocket is the platform independent base upon which the higher level | 
					
						
							|  |  |  |  |  * network primitives are based. Applications are not typically meant to | 
					
						
							| 
									
										
										
										
											2009-05-15 21:27:54 +02:00
										 |  |  |  |  * use it directly, but rather through classes like #GSocketClient, | 
					
						
							|  |  |  |  |  * #GSocketService and #GSocketConnection. However there may be cases where | 
					
						
							|  |  |  |  |  * direct use of #GSocket is useful. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * #GSocket implements the #GInitable interface, so if it is manually constructed | 
					
						
							|  |  |  |  |  * by e.g. g_object_new() you must call g_initable_init() and check the | 
					
						
							|  |  |  |  |  * results before using the object. This is done automatically in | 
					
						
							|  |  |  |  |  * g_socket_new() and g_socket_new_from_fd(), so these functions can return | 
					
						
							|  |  |  |  |  * %NULL. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sockets operate in two general modes, blocking or non-blocking. When | 
					
						
							| 
									
										
										
										
											2015-07-29 14:08:31 +01:00
										 |  |  |  |  * in blocking mode all operations (which don’t take an explicit blocking | 
					
						
							|  |  |  |  |  * parameter) block until the requested operation | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * is finished or there is an error. In non-blocking mode all calls that | 
					
						
							|  |  |  |  |  * would block return immediately with a %G_IO_ERROR_WOULD_BLOCK error. | 
					
						
							|  |  |  |  |  * To know when a call would successfully run you can call g_socket_condition_check(), | 
					
						
							|  |  |  |  |  * or g_socket_condition_wait(). You can also use g_socket_create_source() and | 
					
						
							|  |  |  |  |  * attach it to a #GMainContext to get callbacks when I/O is possible. | 
					
						
							|  |  |  |  |  * Note that all sockets are always set to non blocking mode in the system, and | 
					
						
							|  |  |  |  |  * blocking mode is emulated in GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * When working in non-blocking mode applications should always be able to | 
					
						
							|  |  |  |  |  * handle getting a %G_IO_ERROR_WOULD_BLOCK error even when some other | 
					
						
							|  |  |  |  |  * function said that I/O was possible. This can easily happen in case | 
					
						
							|  |  |  |  |  * of a race condition in the application, but it can also happen for other | 
					
						
							|  |  |  |  |  * reasons. For instance, on Windows a socket is always seen as writable | 
					
						
							|  |  |  |  |  * until a write returns %G_IO_ERROR_WOULD_BLOCK. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-02-01 11:57:13 -05:00
										 |  |  |  |  * #GSockets can be either connection oriented or datagram based. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * For connection oriented types you must first establish a connection by | 
					
						
							|  |  |  |  |  * either connecting to an address or accepting a connection from another | 
					
						
							|  |  |  |  |  * address. For connectionless socket types the target/source address is | 
					
						
							|  |  |  |  |  * specified or received in each I/O operation. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * All socket file descriptors are set to be close-on-exec. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-17 20:44:54 -04:00
										 |  |  |  |  * Note that creating a #GSocket causes the signal %SIGPIPE to be | 
					
						
							|  |  |  |  |  * ignored for the remainder of the program. If you are writing a | 
					
						
							|  |  |  |  |  * command-line utility that uses #GSocket, you may need to take into | 
					
						
							|  |  |  |  |  * account the fact that your program will not automatically be killed | 
					
						
							|  |  |  |  |  * if it tries to write to %stdout after it has been closed. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-04-16 19:03:56 +01:00
										 |  |  |  |  * Like most other APIs in GLib, #GSocket is not inherently thread safe. To use | 
					
						
							|  |  |  |  |  * a #GSocket concurrently from multiple threads, you must implement your own | 
					
						
							|  |  |  |  |  * locking. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static void     g_socket_initable_iface_init (GInitableIface  *iface); | 
					
						
							|  |  |  |  | static gboolean g_socket_initable_init       (GInitable       *initable, | 
					
						
							|  |  |  |  | 					      GCancellable    *cancellable, | 
					
						
							|  |  |  |  | 					      GError         **error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:50:42 +01:00
										 |  |  |  | static void     g_socket_datagram_based_iface_init       (GDatagramBasedInterface *iface); | 
					
						
							|  |  |  |  | static gint     g_socket_datagram_based_receive_messages (GDatagramBased  *self, | 
					
						
							|  |  |  |  |                                                           GInputMessage   *messages, | 
					
						
							|  |  |  |  |                                                           guint            num_messages, | 
					
						
							|  |  |  |  |                                                           gint             flags, | 
					
						
							|  |  |  |  |                                                           gint64           timeout, | 
					
						
							|  |  |  |  |                                                           GCancellable    *cancellable, | 
					
						
							|  |  |  |  |                                                           GError         **error); | 
					
						
							|  |  |  |  | static gint     g_socket_datagram_based_send_messages    (GDatagramBased  *self, | 
					
						
							|  |  |  |  |                                                           GOutputMessage  *messages, | 
					
						
							|  |  |  |  |                                                           guint            num_messages, | 
					
						
							|  |  |  |  |                                                           gint             flags, | 
					
						
							|  |  |  |  |                                                           gint64           timeout, | 
					
						
							|  |  |  |  |                                                           GCancellable    *cancellable, | 
					
						
							|  |  |  |  |                                                           GError         **error); | 
					
						
							|  |  |  |  | static GSource *g_socket_datagram_based_create_source    (GDatagramBased           *self, | 
					
						
							|  |  |  |  |                                                           GIOCondition              condition, | 
					
						
							|  |  |  |  |                                                           GCancellable             *cancellable); | 
					
						
							|  |  |  |  | static GIOCondition g_socket_datagram_based_condition_check      (GDatagramBased   *datagram_based, | 
					
						
							|  |  |  |  |                                                                   GIOCondition      condition); | 
					
						
							|  |  |  |  | static gboolean     g_socket_datagram_based_condition_wait       (GDatagramBased   *datagram_based, | 
					
						
							|  |  |  |  |                                                                   GIOCondition      condition, | 
					
						
							|  |  |  |  |                                                                   gint64            timeout, | 
					
						
							|  |  |  |  |                                                                   GCancellable     *cancellable, | 
					
						
							|  |  |  |  |                                                                   GError          **error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  | static GSocketAddress * | 
					
						
							|  |  |  |  | cache_recv_address (GSocket *socket, struct sockaddr *native, int native_len); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  | static gssize | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  | g_socket_receive_message_with_timeout  (GSocket                 *socket, | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |                                         GSocketAddress         **address, | 
					
						
							|  |  |  |  |                                         GInputVector            *vectors, | 
					
						
							|  |  |  |  |                                         gint                     num_vectors, | 
					
						
							|  |  |  |  |                                         GSocketControlMessage ***messages, | 
					
						
							|  |  |  |  |                                         gint                    *num_messages, | 
					
						
							|  |  |  |  |                                         gint                    *flags, | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                                         gint64                   timeout, | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |                                         GCancellable            *cancellable, | 
					
						
							|  |  |  |  |                                         GError                 **error); | 
					
						
							| 
									
										
										
										
											2015-06-12 08:32:11 +01:00
										 |  |  |  | static gint | 
					
						
							| 
									
										
										
										
											2015-06-12 08:47:37 +01:00
										 |  |  |  | g_socket_receive_messages_with_timeout (GSocket        *socket, | 
					
						
							|  |  |  |  |                                         GInputMessage  *messages, | 
					
						
							|  |  |  |  |                                         guint           num_messages, | 
					
						
							|  |  |  |  |                                         gint            flags, | 
					
						
							|  |  |  |  |                                         gint64          timeout, | 
					
						
							|  |  |  |  |                                         GCancellable   *cancellable, | 
					
						
							|  |  |  |  |                                         GError        **error); | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  | static gssize | 
					
						
							|  |  |  |  | g_socket_send_message_with_timeout     (GSocket                 *socket, | 
					
						
							|  |  |  |  |                                         GSocketAddress          *address, | 
					
						
							|  |  |  |  |                                         GOutputVector           *vectors, | 
					
						
							|  |  |  |  |                                         gint                     num_vectors, | 
					
						
							|  |  |  |  |                                         GSocketControlMessage  **messages, | 
					
						
							|  |  |  |  |                                         gint                     num_messages, | 
					
						
							|  |  |  |  |                                         gint                     flags, | 
					
						
							|  |  |  |  |                                         gint64                   timeout, | 
					
						
							|  |  |  |  |                                         GCancellable            *cancellable, | 
					
						
							|  |  |  |  |                                         GError                 **error); | 
					
						
							| 
									
										
										
										
											2015-06-12 08:47:37 +01:00
										 |  |  |  | static gint | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  | g_socket_send_messages_with_timeout    (GSocket        *socket, | 
					
						
							| 
									
										
										
										
											2015-06-12 08:32:11 +01:00
										 |  |  |  |                                         GOutputMessage *messages, | 
					
						
							|  |  |  |  |                                         guint           num_messages, | 
					
						
							|  |  |  |  |                                         gint            flags, | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                                         gint64          timeout, | 
					
						
							| 
									
										
										
										
											2015-06-12 08:32:11 +01:00
										 |  |  |  |                                         GCancellable   *cancellable, | 
					
						
							|  |  |  |  |                                         GError        **error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | enum | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   PROP_0, | 
					
						
							|  |  |  |  |   PROP_FAMILY, | 
					
						
							|  |  |  |  |   PROP_TYPE, | 
					
						
							|  |  |  |  |   PROP_PROTOCOL, | 
					
						
							|  |  |  |  |   PROP_FD, | 
					
						
							|  |  |  |  |   PROP_BLOCKING, | 
					
						
							|  |  |  |  |   PROP_LISTEN_BACKLOG, | 
					
						
							|  |  |  |  |   PROP_KEEPALIVE, | 
					
						
							|  |  |  |  |   PROP_LOCAL_ADDRESS, | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |   PROP_REMOTE_ADDRESS, | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |   PROP_TIMEOUT, | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |   PROP_TTL, | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |   PROP_BROADCAST, | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |   PROP_MULTICAST_LOOPBACK, | 
					
						
							|  |  |  |  |   PROP_MULTICAST_TTL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-09 15:28:36 +01:00
										 |  |  |  | /* Size of the receiver cache for g_socket_receive_from() */ | 
					
						
							|  |  |  |  | #define RECV_ADDR_CACHE_SIZE 8
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | struct _GSocketPrivate | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocketFamily   family; | 
					
						
							|  |  |  |  |   GSocketType     type; | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  |   GSocketProtocol protocol; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   gint            fd; | 
					
						
							|  |  |  |  |   gint            listen_backlog; | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |   guint           timeout; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   GError         *construct_error; | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |   GSocketAddress *remote_address; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   guint           inited : 1; | 
					
						
							|  |  |  |  |   guint           blocking : 1; | 
					
						
							|  |  |  |  |   guint           keepalive : 1; | 
					
						
							|  |  |  |  |   guint           closed : 1; | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |   guint           connected_read : 1; | 
					
						
							|  |  |  |  |   guint           connected_write : 1; | 
					
						
							| 
									
										
										
										
											2009-05-19 10:47:55 +02:00
										 |  |  |  |   guint           listening : 1; | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |   guint           timed_out : 1; | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |   guint           connect_pending : 1; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   WSAEVENT        event; | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |   gboolean        waiting; | 
					
						
							|  |  |  |  |   DWORD           waiting_result; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   int             current_events; | 
					
						
							|  |  |  |  |   int             current_errors; | 
					
						
							|  |  |  |  |   int             selected_events; | 
					
						
							|  |  |  |  |   GList          *requested_conditions; /* list of requested GIOCondition * */ | 
					
						
							| 
									
										
										
										
											2013-07-28 16:43:44 +02:00
										 |  |  |  |   GMutex          win32_source_lock; | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |   GCond           win32_source_cond; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-11-09 15:28:36 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   struct { | 
					
						
							|  |  |  |  |     GSocketAddress *addr; | 
					
						
							|  |  |  |  |     struct sockaddr *native; | 
					
						
							|  |  |  |  |     gint native_len; | 
					
						
							|  |  |  |  |     guint64 last_used; | 
					
						
							|  |  |  |  |   } recv_addr_cache[RECV_ADDR_CACHE_SIZE]; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-11 00:29:58 +01:00
										 |  |  |  | G_DEFINE_TYPE_WITH_CODE (GSocket, g_socket, G_TYPE_OBJECT, | 
					
						
							|  |  |  |  |                          G_ADD_PRIVATE (GSocket) | 
					
						
							|  |  |  |  | 			 g_networking_init (); | 
					
						
							|  |  |  |  | 			 G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE, | 
					
						
							| 
									
										
										
										
											2015-06-12 08:50:42 +01:00
										 |  |  |  | 						g_socket_initable_iface_init); | 
					
						
							|  |  |  |  |                          G_IMPLEMENT_INTERFACE (G_TYPE_DATAGRAM_BASED, | 
					
						
							|  |  |  |  |                                                 g_socket_datagram_based_iface_init)); | 
					
						
							| 
									
										
										
										
											2013-06-11 00:29:58 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | static int | 
					
						
							|  |  |  |  | get_socket_errno (void) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   return errno; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   return WSAGetLastError (); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GIOErrorEnum | 
					
						
							|  |  |  |  | socket_io_error_from_errno (int err) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-03-25 13:52:45 -04:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   return g_io_error_from_win32_error (err); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-03-25 13:52:45 -04:00
										 |  |  |  |   return g_io_error_from_errno (err); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static const char * | 
					
						
							|  |  |  |  | socket_strerror (int err) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   return g_strerror (err); | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-10-04 23:52:38 +08:00
										 |  |  |  |   const char *msg_ret; | 
					
						
							| 
									
										
										
										
											2010-04-12 10:21:20 -04:00
										 |  |  |  |   char *msg; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   msg = g_win32_error_message (err); | 
					
						
							| 
									
										
										
										
											2010-04-12 10:21:20 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-04 23:52:38 +08:00
										 |  |  |  |   msg_ret = g_intern_string (msg); | 
					
						
							|  |  |  |  |   g_free (msg); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return msg_ret; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 10:18:06 -04:00
										 |  |  |  | /* Wrapper around g_set_error() to avoid doing excess work */ | 
					
						
							|  |  |  |  | #define socket_set_error_lazy(err, errsv, fmt)                          \
 | 
					
						
							|  |  |  |  |   G_STMT_START {                                                        \ | 
					
						
							|  |  |  |  |     GError **__err = (err);                                             \ | 
					
						
							|  |  |  |  |     int __errsv = (errsv);                                              \ | 
					
						
							|  |  |  |  |                                                                         \ | 
					
						
							|  |  |  |  |     if (__err)                                                          \ | 
					
						
							|  |  |  |  |       {                                                                 \ | 
					
						
							|  |  |  |  |         int __code = socket_io_error_from_errno (__errsv);              \ | 
					
						
							|  |  |  |  |         const char *__strerr = socket_strerror (__errsv);               \ | 
					
						
							|  |  |  |  |                                                                         \ | 
					
						
							|  |  |  |  |         if (__code == G_IO_ERROR_WOULD_BLOCK)                           \ | 
					
						
							|  |  |  |  |           g_set_error_literal (__err, G_IO_ERROR, __code, __strerr);    \ | 
					
						
							|  |  |  |  |         else                                                            \ | 
					
						
							|  |  |  |  |           g_set_error (__err, G_IO_ERROR, __code, fmt, __strerr);       \ | 
					
						
							|  |  |  |  |       }                                                                 \ | 
					
						
							|  |  |  |  |   } G_STMT_END | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  | #define win32_unset_event_mask(_socket, _mask) _win32_unset_event_mask (_socket, _mask)
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | _win32_unset_event_mask (GSocket *socket, int mask) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |   g_mutex_lock (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   socket->priv->current_events &= ~mask; | 
					
						
							|  |  |  |  |   socket->priv->current_errors &= ~mask; | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |   g_mutex_unlock (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  | #define win32_unset_event_mask(_socket, _mask)
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-11 13:05:16 -05:00
										 |  |  |  | /* Windows has broken prototypes... */ | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  | #define getsockopt(sockfd, level, optname, optval, optlen) \
 | 
					
						
							|  |  |  |  |   getsockopt (sockfd, level, optname, (gpointer) optval, (int*) optlen) | 
					
						
							|  |  |  |  | #define setsockopt(sockfd, level, optname, optval, optlen) \
 | 
					
						
							|  |  |  |  |   setsockopt (sockfd, level, optname, (gpointer) optval, optlen) | 
					
						
							|  |  |  |  | #define getsockname(sockfd, addr, addrlen) \
 | 
					
						
							|  |  |  |  |   getsockname (sockfd, addr, (int *)addrlen) | 
					
						
							|  |  |  |  | #define getpeername(sockfd, addr, addrlen) \
 | 
					
						
							|  |  |  |  |   getpeername (sockfd, addr, (int *)addrlen) | 
					
						
							|  |  |  |  | #define recv(sockfd, buf, len, flags) \
 | 
					
						
							|  |  |  |  |   recv (sockfd, (gpointer)buf, len, flags) | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | static gboolean | 
					
						
							|  |  |  |  | check_socket (GSocket *socket, | 
					
						
							|  |  |  |  | 	      GError **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!socket->priv->inited) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-05-19 15:03:14 +02:00
										 |  |  |  |       g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NOT_INITIALIZED, | 
					
						
							|  |  |  |  |                            _("Invalid socket, not initialized")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->construct_error) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_INITIALIZED, | 
					
						
							|  |  |  |  | 		   _("Invalid socket, initialization failed due to: %s"), | 
					
						
							|  |  |  |  | 		   socket->priv->construct_error->message); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->closed) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_CLOSED, | 
					
						
							|  |  |  |  | 			   _("Socket is already closed")); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-14 17:46:38 -04:00
										 |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							|  |  |  |  | check_timeout (GSocket *socket, | 
					
						
							|  |  |  |  | 	       GError **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |   if (socket->priv->timed_out) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       socket->priv->timed_out = FALSE; | 
					
						
							|  |  |  |  |       g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_TIMED_OUT, | 
					
						
							|  |  |  |  | 			   _("Socket I/O timed out")); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_details_from_fd (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   struct sockaddr_storage address; | 
					
						
							|  |  |  |  |   gint fd; | 
					
						
							|  |  |  |  |   guint addrlen; | 
					
						
							| 
									
										
										
										
											2011-09-03 19:14:16 -04:00
										 |  |  |  |   int value, family; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   int errsv; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   fd = socket->priv->fd; | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (!g_socket_get_option (socket, SOL_SOCKET, SO_TYPE, &value, NULL)) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       errsv = get_socket_errno (); | 
					
						
							|  |  |  |  |       goto err; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   switch (value) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |      case SOCK_STREAM: | 
					
						
							|  |  |  |  |       socket->priv->type = G_SOCKET_TYPE_STREAM; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |      case SOCK_DGRAM: | 
					
						
							|  |  |  |  |       socket->priv->type = G_SOCKET_TYPE_DATAGRAM; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |      case SOCK_SEQPACKET: | 
					
						
							|  |  |  |  |       socket->priv->type = G_SOCKET_TYPE_SEQPACKET; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |      default: | 
					
						
							|  |  |  |  |       socket->priv->type = G_SOCKET_TYPE_INVALID; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   addrlen = sizeof address; | 
					
						
							|  |  |  |  |   if (getsockname (fd, (struct sockaddr *) &address, &addrlen) != 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       errsv = get_socket_errno (); | 
					
						
							|  |  |  |  |       goto err; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-03 19:14:16 -04:00
										 |  |  |  |   if (addrlen > 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_assert (G_STRUCT_OFFSET (struct sockaddr, sa_family) + | 
					
						
							|  |  |  |  | 		sizeof address.ss_family <= addrlen); | 
					
						
							|  |  |  |  |       family = address.ss_family; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       /* On Solaris, this happens if the socket is not yet connected.
 | 
					
						
							|  |  |  |  |        * But we can use SO_DOMAIN as a workaround there. | 
					
						
							|  |  |  |  |        */ | 
					
						
							|  |  |  |  | #ifdef SO_DOMAIN
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       if (!g_socket_get_option (socket, SOL_SOCKET, SO_DOMAIN, &family, NULL)) | 
					
						
							| 
									
										
										
										
											2011-09-03 19:14:16 -04:00
										 |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 	  goto err; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-09-04 08:42:31 -04:00
										 |  |  |  |       /* This will translate to G_IO_ERROR_FAILED on either unix or windows */ | 
					
						
							|  |  |  |  |       errsv = -1; | 
					
						
							| 
									
										
										
										
											2011-09-03 19:14:16 -04:00
										 |  |  |  |       goto err; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   switch (family) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |      case G_SOCKET_FAMILY_IPV4: | 
					
						
							|  |  |  |  |      case G_SOCKET_FAMILY_IPV6: | 
					
						
							| 
									
										
										
										
											2010-08-17 18:34:13 -04:00
										 |  |  |  |        socket->priv->family = address.ss_family; | 
					
						
							|  |  |  |  |        switch (socket->priv->type) | 
					
						
							|  |  |  |  | 	 { | 
					
						
							|  |  |  |  | 	 case G_SOCKET_TYPE_STREAM: | 
					
						
							|  |  |  |  | 	   socket->priv->protocol = G_SOCKET_PROTOCOL_TCP; | 
					
						
							|  |  |  |  | 	   break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	 case G_SOCKET_TYPE_DATAGRAM: | 
					
						
							|  |  |  |  | 	   socket->priv->protocol = G_SOCKET_PROTOCOL_UDP; | 
					
						
							|  |  |  |  | 	   break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	 case G_SOCKET_TYPE_SEQPACKET: | 
					
						
							|  |  |  |  | 	   socket->priv->protocol = G_SOCKET_PROTOCOL_SCTP; | 
					
						
							|  |  |  |  | 	   break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	 default: | 
					
						
							|  |  |  |  | 	   break; | 
					
						
							|  |  |  |  | 	 } | 
					
						
							|  |  |  |  |        break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |      case G_SOCKET_FAMILY_UNIX: | 
					
						
							| 
									
										
										
										
											2010-08-17 18:34:13 -04:00
										 |  |  |  |        socket->priv->family = G_SOCKET_FAMILY_UNIX; | 
					
						
							|  |  |  |  |        socket->priv->protocol = G_SOCKET_PROTOCOL_DEFAULT; | 
					
						
							|  |  |  |  |        break; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |      default: | 
					
						
							| 
									
										
										
										
											2010-08-17 18:34:13 -04:00
										 |  |  |  |        socket->priv->family = G_SOCKET_FAMILY_INVALID; | 
					
						
							|  |  |  |  |        break; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->family != G_SOCKET_FAMILY_INVALID) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       addrlen = sizeof address; | 
					
						
							|  |  |  |  |       if (getpeername (fd, (struct sockaddr *) &address, &addrlen) >= 0) | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |         { | 
					
						
							|  |  |  |  |           socket->priv->connected_read = TRUE; | 
					
						
							|  |  |  |  |           socket->priv->connected_write = TRUE; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (g_socket_get_option (socket, SOL_SOCKET, SO_KEEPALIVE, &value, NULL)) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       socket->priv->keepalive = !!value; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       /* Can't read, maybe not supported, assume FALSE */ | 
					
						
							|  |  |  |  |       socket->priv->keepalive = FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |  err: | 
					
						
							|  |  |  |  |   g_set_error (&socket->priv->construct_error, G_IO_ERROR, | 
					
						
							|  |  |  |  | 	       socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 	       _("creating GSocket from fd: %s"), | 
					
						
							|  |  |  |  | 	       socket_strerror (errsv)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 16:39:49 -05:00
										 |  |  |  | /* Wrapper around socket() that is shared with gnetworkmonitornetlink.c */ | 
					
						
							|  |  |  |  | gint | 
					
						
							|  |  |  |  | g_socket (gint     domain, | 
					
						
							|  |  |  |  |           gint     type, | 
					
						
							|  |  |  |  |           gint     protocol, | 
					
						
							|  |  |  |  |           GError **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   int fd; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef SOCK_CLOEXEC
 | 
					
						
							|  |  |  |  |   fd = socket (domain, type | SOCK_CLOEXEC, protocol); | 
					
						
							|  |  |  |  |   if (fd != -1) | 
					
						
							|  |  |  |  |     return fd; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* It's possible that libc has SOCK_CLOEXEC but the kernel does not */ | 
					
						
							| 
									
										
										
										
											2013-09-18 16:39:09 +02:00
										 |  |  |  |   if (fd < 0 && (errno == EINVAL || errno == EPROTOTYPE)) | 
					
						
							| 
									
										
										
										
											2013-01-22 16:39:49 -05:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  |     fd = socket (domain, type, protocol); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (fd < 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 		   _("Unable to create socket: %s"), socket_strerror (errsv)); | 
					
						
							|  |  |  |  |       errno = errsv; | 
					
						
							|  |  |  |  |       return -1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     int flags; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* We always want to set close-on-exec to protect users. If you
 | 
					
						
							|  |  |  |  |        need to so some weird inheritance to exec you can re-enable this | 
					
						
							|  |  |  |  |        using lower level hacks with g_socket_get_fd(). */ | 
					
						
							|  |  |  |  |     flags = fcntl (fd, F_GETFD, 0); | 
					
						
							|  |  |  |  |     if (flags != -1 && | 
					
						
							|  |  |  |  | 	(flags & FD_CLOEXEC) == 0) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	flags |= FD_CLOEXEC; | 
					
						
							|  |  |  |  | 	fcntl (fd, F_SETFD, flags); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return fd; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | static gint | 
					
						
							|  |  |  |  | g_socket_create_socket (GSocketFamily   family, | 
					
						
							|  |  |  |  | 			GSocketType     type, | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  | 			int             protocol, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 			GError        **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint native_type; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   switch (type) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |      case G_SOCKET_TYPE_STREAM: | 
					
						
							|  |  |  |  |       native_type = SOCK_STREAM; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |      case G_SOCKET_TYPE_DATAGRAM: | 
					
						
							|  |  |  |  |       native_type = SOCK_DGRAM; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |      case G_SOCKET_TYPE_SEQPACKET: | 
					
						
							|  |  |  |  |       native_type = SOCK_SEQPACKET; | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |      default: | 
					
						
							|  |  |  |  |       g_assert_not_reached (); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-25 10:44:55 -04:00
										 |  |  |  |   if (family <= 0) | 
					
						
							| 
									
										
										
										
											2012-04-04 11:39:03 +02:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT, | 
					
						
							|  |  |  |  |                    _("Unable to create socket: %s"), _("Unknown family was specified")); | 
					
						
							|  |  |  |  |       return -1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  |   if (protocol == -1) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-05-15 09:10:23 +02:00
										 |  |  |  |       g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT, | 
					
						
							|  |  |  |  | 		   _("Unable to create socket: %s"), _("Unknown protocol was specified")); | 
					
						
							|  |  |  |  |       return -1; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 16:39:49 -05:00
										 |  |  |  |   return g_socket (family, native_type, protocol, error); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_constructed (GObject *object) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocket *socket = G_SOCKET (object); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->fd >= 0) | 
					
						
							|  |  |  |  |     /* create socket->priv info from the fd */ | 
					
						
							|  |  |  |  |     g_socket_details_from_fd (socket); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     /* create the fd from socket->priv info */ | 
					
						
							|  |  |  |  |     socket->priv->fd = g_socket_create_socket (socket->priv->family, | 
					
						
							|  |  |  |  | 					       socket->priv->type, | 
					
						
							|  |  |  |  | 					       socket->priv->protocol, | 
					
						
							|  |  |  |  | 					       &socket->priv->construct_error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->fd != -1) | 
					
						
							| 
									
										
										
										
											2014-05-01 09:58:07 -04:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |       GError *error = NULL; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |       gulong arg; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       /* Always use native nonblocking sockets, as Windows sets sockets to
 | 
					
						
							|  |  |  |  |        * nonblocking automatically in certain operations. This way we make | 
					
						
							|  |  |  |  |        * things work the same on all platforms. | 
					
						
							|  |  |  |  |        */ | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |       if (!g_unix_set_fd_nonblocking (socket->priv->fd, TRUE, &error)) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_warning ("Error setting socket nonblocking: %s", error->message); | 
					
						
							|  |  |  |  |           g_clear_error (&error); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |       arg = TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (ioctlsocket (socket->priv->fd, FIONBIO, &arg) == SOCKET_ERROR) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  |           g_warning ("Error setting socket status flags: %s", socket_strerror (errsv)); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-05-01 09:59:05 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef SO_NOSIGPIPE
 | 
					
						
							|  |  |  |  |       /* See note about SIGPIPE below. */ | 
					
						
							|  |  |  |  |       g_socket_set_option (socket, SOL_SOCKET, SO_NOSIGPIPE, TRUE, NULL); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-05-01 09:58:07 -04:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_get_property (GObject    *object, | 
					
						
							|  |  |  |  | 		       guint       prop_id, | 
					
						
							|  |  |  |  | 		       GValue     *value, | 
					
						
							|  |  |  |  | 		       GParamSpec *pspec) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocket *socket = G_SOCKET (object); | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   GSocketAddress *address; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   switch (prop_id) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       case PROP_FAMILY: | 
					
						
							|  |  |  |  | 	g_value_set_enum (value, socket->priv->family); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_TYPE: | 
					
						
							|  |  |  |  | 	g_value_set_enum (value, socket->priv->type); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_PROTOCOL: | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  | 	g_value_set_enum (value, socket->priv->protocol); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_FD: | 
					
						
							|  |  |  |  | 	g_value_set_int (value, socket->priv->fd); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_BLOCKING: | 
					
						
							|  |  |  |  | 	g_value_set_boolean (value, socket->priv->blocking); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_LISTEN_BACKLOG: | 
					
						
							|  |  |  |  | 	g_value_set_int (value, socket->priv->listen_backlog); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_KEEPALIVE: | 
					
						
							|  |  |  |  | 	g_value_set_boolean (value, socket->priv->keepalive); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_LOCAL_ADDRESS: | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  | 	address = g_socket_get_local_address (socket, NULL); | 
					
						
							|  |  |  |  | 	g_value_take_object (value, address); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_REMOTE_ADDRESS: | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  | 	address = g_socket_get_remote_address (socket, NULL); | 
					
						
							|  |  |  |  | 	g_value_take_object (value, address); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |       case PROP_TIMEOUT: | 
					
						
							|  |  |  |  | 	g_value_set_uint (value, socket->priv->timeout); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |       case PROP_TTL: | 
					
						
							|  |  |  |  | 	g_value_set_uint (value, g_socket_get_ttl (socket)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |       case PROP_BROADCAST: | 
					
						
							|  |  |  |  | 	g_value_set_boolean (value, g_socket_get_broadcast (socket)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       case PROP_MULTICAST_LOOPBACK: | 
					
						
							|  |  |  |  | 	g_value_set_boolean (value, g_socket_get_multicast_loopback (socket)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_MULTICAST_TTL: | 
					
						
							|  |  |  |  | 	g_value_set_uint (value, g_socket_get_multicast_ttl (socket)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       default: | 
					
						
							|  |  |  |  | 	G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_set_property (GObject      *object, | 
					
						
							|  |  |  |  | 		       guint         prop_id, | 
					
						
							|  |  |  |  | 		       const GValue *value, | 
					
						
							|  |  |  |  | 		       GParamSpec   *pspec) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocket *socket = G_SOCKET (object); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   switch (prop_id) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       case PROP_FAMILY: | 
					
						
							|  |  |  |  | 	socket->priv->family = g_value_get_enum (value); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_TYPE: | 
					
						
							|  |  |  |  | 	socket->priv->type = g_value_get_enum (value); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_PROTOCOL: | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  | 	socket->priv->protocol = g_value_get_enum (value); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_FD: | 
					
						
							|  |  |  |  | 	socket->priv->fd = g_value_get_int (value); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_BLOCKING: | 
					
						
							|  |  |  |  | 	g_socket_set_blocking (socket, g_value_get_boolean (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_LISTEN_BACKLOG: | 
					
						
							|  |  |  |  | 	g_socket_set_listen_backlog (socket, g_value_get_int (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_KEEPALIVE: | 
					
						
							|  |  |  |  | 	g_socket_set_keepalive (socket, g_value_get_boolean (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |       case PROP_TIMEOUT: | 
					
						
							|  |  |  |  | 	g_socket_set_timeout (socket, g_value_get_uint (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |       case PROP_TTL: | 
					
						
							|  |  |  |  | 	g_socket_set_ttl (socket, g_value_get_uint (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |       case PROP_BROADCAST: | 
					
						
							|  |  |  |  | 	g_socket_set_broadcast (socket, g_value_get_boolean (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       case PROP_MULTICAST_LOOPBACK: | 
					
						
							|  |  |  |  | 	g_socket_set_multicast_loopback (socket, g_value_get_boolean (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       case PROP_MULTICAST_TTL: | 
					
						
							|  |  |  |  | 	g_socket_set_multicast_ttl (socket, g_value_get_uint (value)); | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       default: | 
					
						
							|  |  |  |  | 	G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_finalize (GObject *object) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocket *socket = G_SOCKET (object); | 
					
						
							| 
									
										
										
										
											2012-11-09 15:28:36 +01:00
										 |  |  |  |   gint i; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_clear_error (&socket->priv->construct_error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->fd != -1 && | 
					
						
							|  |  |  |  |       !socket->priv->closed) | 
					
						
							|  |  |  |  |     g_socket_close (socket, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |   if (socket->priv->remote_address) | 
					
						
							|  |  |  |  |     g_object_unref (socket->priv->remote_address); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							| 
									
										
										
										
											2010-03-14 11:47:17 +02:00
										 |  |  |  |   if (socket->priv->event != WSA_INVALID_EVENT) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       WSACloseEvent (socket->priv->event); | 
					
						
							|  |  |  |  |       socket->priv->event = WSA_INVALID_EVENT; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   g_assert (socket->priv->requested_conditions == NULL); | 
					
						
							| 
									
										
										
										
											2013-07-28 16:43:44 +02:00
										 |  |  |  |   g_mutex_clear (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |   g_cond_clear (&socket->priv->win32_source_cond); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-09 15:28:36 +01:00
										 |  |  |  |   for (i = 0; i < RECV_ADDR_CACHE_SIZE; i++) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (socket->priv->recv_addr_cache[i].addr) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_object_unref (socket->priv->recv_addr_cache[i].addr); | 
					
						
							|  |  |  |  |           g_free (socket->priv->recv_addr_cache[i].native); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   if (G_OBJECT_CLASS (g_socket_parent_class)->finalize) | 
					
						
							|  |  |  |  |     (*G_OBJECT_CLASS (g_socket_parent_class)->finalize) (object); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_class_init (GSocketClass *klass) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GObjectClass *gobject_class G_GNUC_UNUSED = G_OBJECT_CLASS (klass); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-17 20:44:54 -04:00
										 |  |  |  | #ifdef SIGPIPE
 | 
					
						
							|  |  |  |  |   /* There is no portable, thread-safe way to avoid having the process
 | 
					
						
							|  |  |  |  |    * be killed by SIGPIPE when calling send() or sendmsg(), so we are | 
					
						
							|  |  |  |  |    * forced to simply ignore the signal process-wide. | 
					
						
							| 
									
										
										
										
											2014-05-01 09:59:05 -04:00
										 |  |  |  |    * | 
					
						
							|  |  |  |  |    * Even if we ignore it though, gdb will still stop if the app | 
					
						
							|  |  |  |  |    * receives a SIGPIPE, which can be confusing and annoying. So when | 
					
						
							|  |  |  |  |    * possible, we also use MSG_NOSIGNAL / SO_NOSIGPIPE elsewhere to | 
					
						
							|  |  |  |  |    * prevent the signal from occurring at all. | 
					
						
							| 
									
										
										
										
											2009-05-17 20:44:54 -04:00
										 |  |  |  |    */ | 
					
						
							|  |  |  |  |   signal (SIGPIPE, SIG_IGN); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   gobject_class->finalize = g_socket_finalize; | 
					
						
							|  |  |  |  |   gobject_class->constructed = g_socket_constructed; | 
					
						
							|  |  |  |  |   gobject_class->set_property = g_socket_set_property; | 
					
						
							|  |  |  |  |   gobject_class->get_property = g_socket_get_property; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_FAMILY, | 
					
						
							|  |  |  |  | 				   g_param_spec_enum ("family", | 
					
						
							|  |  |  |  | 						      P_("Socket family"), | 
					
						
							|  |  |  |  | 						      P_("The sockets address family"), | 
					
						
							|  |  |  |  | 						      G_TYPE_SOCKET_FAMILY, | 
					
						
							|  |  |  |  | 						      G_SOCKET_FAMILY_INVALID, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 						      G_PARAM_CONSTRUCT_ONLY | | 
					
						
							|  |  |  |  |                                                       G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                       G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_TYPE, | 
					
						
							|  |  |  |  | 				   g_param_spec_enum ("type", | 
					
						
							|  |  |  |  | 						      P_("Socket type"), | 
					
						
							|  |  |  |  | 						      P_("The sockets type"), | 
					
						
							|  |  |  |  | 						      G_TYPE_SOCKET_TYPE, | 
					
						
							|  |  |  |  | 						      G_SOCKET_TYPE_STREAM, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 						      G_PARAM_CONSTRUCT_ONLY | | 
					
						
							|  |  |  |  |                                                       G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                       G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_PROTOCOL, | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  | 				   g_param_spec_enum ("protocol", | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 						      P_("Socket protocol"), | 
					
						
							|  |  |  |  | 						      P_("The id of the protocol to use, or -1 for unknown"), | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  | 						      G_TYPE_SOCKET_PROTOCOL, | 
					
						
							|  |  |  |  | 						      G_SOCKET_PROTOCOL_UNKNOWN, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 						      G_PARAM_CONSTRUCT_ONLY | | 
					
						
							|  |  |  |  |                                                       G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                       G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_FD, | 
					
						
							|  |  |  |  | 				   g_param_spec_int ("fd", | 
					
						
							|  |  |  |  | 						     P_("File descriptor"), | 
					
						
							|  |  |  |  | 						     P_("The sockets file descriptor"), | 
					
						
							|  |  |  |  | 						     G_MININT, | 
					
						
							|  |  |  |  | 						     G_MAXINT, | 
					
						
							|  |  |  |  | 						     -1, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 						     G_PARAM_CONSTRUCT_ONLY | | 
					
						
							|  |  |  |  |                                                      G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                      G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_BLOCKING, | 
					
						
							|  |  |  |  | 				   g_param_spec_boolean ("blocking", | 
					
						
							|  |  |  |  | 							 P_("blocking"), | 
					
						
							|  |  |  |  | 							 P_("Whether or not I/O on this socket is blocking"), | 
					
						
							|  |  |  |  | 							 TRUE, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 							 G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                          G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_LISTEN_BACKLOG, | 
					
						
							|  |  |  |  | 				   g_param_spec_int ("listen-backlog", | 
					
						
							|  |  |  |  | 						     P_("Listen backlog"), | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 						     P_("Outstanding connections in the listen queue"), | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 						     0, | 
					
						
							|  |  |  |  | 						     SOMAXCONN, | 
					
						
							|  |  |  |  | 						     10, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 						     G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                      G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_KEEPALIVE, | 
					
						
							|  |  |  |  | 				   g_param_spec_boolean ("keepalive", | 
					
						
							|  |  |  |  | 							 P_("Keep connection alive"), | 
					
						
							|  |  |  |  | 							 P_("Keep connection alive by sending periodic pings"), | 
					
						
							|  |  |  |  | 							 FALSE, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 							 G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                          G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_LOCAL_ADDRESS, | 
					
						
							|  |  |  |  | 				   g_param_spec_object ("local-address", | 
					
						
							|  |  |  |  | 							P_("Local address"), | 
					
						
							|  |  |  |  | 							P_("The local address the socket is bound to"), | 
					
						
							|  |  |  |  | 							G_TYPE_SOCKET_ADDRESS, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 							G_PARAM_READABLE | | 
					
						
							|  |  |  |  |                                                         G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_REMOTE_ADDRESS, | 
					
						
							|  |  |  |  | 				   g_param_spec_object ("remote-address", | 
					
						
							|  |  |  |  | 							P_("Remote address"), | 
					
						
							|  |  |  |  | 							P_("The remote address the socket is connected to"), | 
					
						
							|  |  |  |  | 							G_TYPE_SOCKET_ADDRESS, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 							G_PARAM_READABLE | | 
					
						
							|  |  |  |  |                                                         G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   /**
 | 
					
						
							|  |  |  |  |    * GSocket:timeout: | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * The timeout in seconds on socket I/O | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * Since: 2.26 | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_TIMEOUT, | 
					
						
							|  |  |  |  | 				   g_param_spec_uint ("timeout", | 
					
						
							|  |  |  |  | 						      P_("Timeout"), | 
					
						
							|  |  |  |  | 						      P_("The timeout in seconds on socket I/O"), | 
					
						
							|  |  |  |  | 						      0, | 
					
						
							|  |  |  |  | 						      G_MAXUINT, | 
					
						
							|  |  |  |  | 						      0, | 
					
						
							|  |  |  |  | 						      G_PARAM_READWRITE | | 
					
						
							|  |  |  |  | 						      G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |   /**
 | 
					
						
							|  |  |  |  |    * GSocket:broadcast: | 
					
						
							|  |  |  |  |    * | 
					
						
							| 
									
										
										
										
											2013-02-17 09:41:55 -05:00
										 |  |  |  |    * Whether the socket should allow sending to broadcast addresses. | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |    * | 
					
						
							|  |  |  |  |    * Since: 2.32 | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_BROADCAST, | 
					
						
							|  |  |  |  | 				   g_param_spec_boolean ("broadcast", | 
					
						
							|  |  |  |  | 							 P_("Broadcast"), | 
					
						
							| 
									
										
										
										
											2013-02-17 09:41:55 -05:00
										 |  |  |  | 							 P_("Whether to allow sending to broadcast addresses"), | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  | 							 FALSE, | 
					
						
							|  |  |  |  | 							 G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                          G_PARAM_STATIC_STRINGS)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |   /**
 | 
					
						
							|  |  |  |  |    * GSocket:ttl: | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * Time-to-live for outgoing unicast packets | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * Since: 2.32 | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_TTL, | 
					
						
							|  |  |  |  | 				   g_param_spec_uint ("ttl", | 
					
						
							|  |  |  |  | 						      P_("TTL"), | 
					
						
							|  |  |  |  | 						      P_("Time-to-live of outgoing unicast packets"), | 
					
						
							|  |  |  |  | 						      0, G_MAXUINT, 0, | 
					
						
							|  |  |  |  | 						      G_PARAM_READWRITE | | 
					
						
							|  |  |  |  | 						      G_PARAM_STATIC_STRINGS)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |   /**
 | 
					
						
							|  |  |  |  |    * GSocket:multicast-loopback: | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * Whether outgoing multicast packets loop back to the local host. | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * Since: 2.32 | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_MULTICAST_LOOPBACK, | 
					
						
							|  |  |  |  | 				   g_param_spec_boolean ("multicast-loopback", | 
					
						
							|  |  |  |  | 							 P_("Multicast loopback"), | 
					
						
							|  |  |  |  | 							 P_("Whether outgoing multicast packets loop back to the local host"), | 
					
						
							|  |  |  |  | 							 TRUE, | 
					
						
							|  |  |  |  | 							 G_PARAM_READWRITE | | 
					
						
							|  |  |  |  |                                                          G_PARAM_STATIC_STRINGS)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /**
 | 
					
						
							|  |  |  |  |    * GSocket:multicast-ttl: | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * Time-to-live out outgoing multicast packets | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * Since: 2.32 | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   g_object_class_install_property (gobject_class, PROP_MULTICAST_TTL, | 
					
						
							|  |  |  |  | 				   g_param_spec_uint ("multicast-ttl", | 
					
						
							|  |  |  |  | 						      P_("Multicast TTL"), | 
					
						
							|  |  |  |  | 						      P_("Time-to-live of outgoing multicast packets"), | 
					
						
							|  |  |  |  | 						      0, G_MAXUINT, 1, | 
					
						
							|  |  |  |  | 						      G_PARAM_READWRITE | | 
					
						
							|  |  |  |  | 						      G_PARAM_STATIC_STRINGS)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_initable_iface_init (GInitableIface *iface) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   iface->init = g_socket_initable_init; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:50:42 +01:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_datagram_based_iface_init (GDatagramBasedInterface *iface) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   iface->receive_messages = g_socket_datagram_based_receive_messages; | 
					
						
							|  |  |  |  |   iface->send_messages = g_socket_datagram_based_send_messages; | 
					
						
							|  |  |  |  |   iface->create_source = g_socket_datagram_based_create_source; | 
					
						
							|  |  |  |  |   iface->condition_check = g_socket_datagram_based_condition_check; | 
					
						
							|  |  |  |  |   iface->condition_wait = g_socket_datagram_based_condition_wait; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | g_socket_init (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-24 15:43:04 +01:00
										 |  |  |  |   socket->priv = g_socket_get_instance_private (socket); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket->priv->fd = -1; | 
					
						
							|  |  |  |  |   socket->priv->blocking = TRUE; | 
					
						
							|  |  |  |  |   socket->priv->listen_backlog = 10; | 
					
						
							|  |  |  |  |   socket->priv->construct_error = NULL; | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   socket->priv->event = WSA_INVALID_EVENT; | 
					
						
							| 
									
										
										
										
											2013-07-28 16:43:44 +02:00
										 |  |  |  |   g_mutex_init (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |   g_cond_init (&socket->priv->win32_source_cond); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							|  |  |  |  | g_socket_initable_init (GInitable *initable, | 
					
						
							|  |  |  |  | 			GCancellable *cancellable, | 
					
						
							|  |  |  |  | 			GError  **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocket  *socket; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (initable), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket = G_SOCKET (initable); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (cancellable != NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-05-19 15:03:14 +02:00
										 |  |  |  |       g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, | 
					
						
							|  |  |  |  |                            _("Cancellable initialization not supported")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket->priv->inited = TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->construct_error) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (error) | 
					
						
							|  |  |  |  | 	*error = g_error_copy (socket->priv->construct_error); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:50:42 +01:00
										 |  |  |  | static gboolean | 
					
						
							|  |  |  |  | check_datagram_based (GDatagramBased  *self, | 
					
						
							|  |  |  |  |                       GError         **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   switch (g_socket_get_socket_type (G_SOCKET (self))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |     case G_SOCKET_TYPE_INVALID: | 
					
						
							|  |  |  |  |     case G_SOCKET_TYPE_STREAM: | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, | 
					
						
							|  |  |  |  |                    _("Cannot use datagram operations on a non-datagram " | 
					
						
							|  |  |  |  |                      "socket.")); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     case G_SOCKET_TYPE_DATAGRAM: | 
					
						
							|  |  |  |  |     case G_SOCKET_TYPE_SEQPACKET: | 
					
						
							|  |  |  |  |       /* Fall through. */ | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* Due to us sharing #GSocketSource with the #GSocket implementation, it is
 | 
					
						
							|  |  |  |  |    * pretty tricky to split out #GSocket:timeout so that it does not affect | 
					
						
							|  |  |  |  |    * #GDatagramBased operations (but still affects #GSocket operations). It is | 
					
						
							|  |  |  |  |    * not worth that effort — just disallow it and require the user to specify | 
					
						
							|  |  |  |  |    * timeouts on a per-operation basis. */ | 
					
						
							|  |  |  |  |   if (g_socket_get_timeout (G_SOCKET (self)) != 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, | 
					
						
							|  |  |  |  |                    _("Cannot use datagram operations on a socket with a " | 
					
						
							|  |  |  |  |                      "timeout set.")); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gint | 
					
						
							|  |  |  |  | g_socket_datagram_based_receive_messages (GDatagramBased  *self, | 
					
						
							|  |  |  |  |                                           GInputMessage   *messages, | 
					
						
							|  |  |  |  |                                           guint            num_messages, | 
					
						
							|  |  |  |  |                                           gint             flags, | 
					
						
							|  |  |  |  |                                           gint64           timeout, | 
					
						
							|  |  |  |  |                                           GCancellable    *cancellable, | 
					
						
							|  |  |  |  |                                           GError         **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!check_datagram_based (self, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_receive_messages_with_timeout (G_SOCKET (self), messages, | 
					
						
							|  |  |  |  |                                                  num_messages, flags, timeout, | 
					
						
							|  |  |  |  |                                                  cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gint | 
					
						
							|  |  |  |  | g_socket_datagram_based_send_messages (GDatagramBased  *self, | 
					
						
							|  |  |  |  |                                        GOutputMessage  *messages, | 
					
						
							|  |  |  |  |                                        guint            num_messages, | 
					
						
							|  |  |  |  |                                        gint             flags, | 
					
						
							|  |  |  |  |                                        gint64           timeout, | 
					
						
							|  |  |  |  |                                        GCancellable    *cancellable, | 
					
						
							|  |  |  |  |                                        GError         **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!check_datagram_based (self, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_send_messages_with_timeout (G_SOCKET (self), messages, | 
					
						
							|  |  |  |  |                                               num_messages, flags, timeout, | 
					
						
							|  |  |  |  |                                               cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GSource * | 
					
						
							|  |  |  |  | g_socket_datagram_based_create_source (GDatagramBased  *self, | 
					
						
							|  |  |  |  |                                        GIOCondition     condition, | 
					
						
							|  |  |  |  |                                        GCancellable    *cancellable) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!check_datagram_based (self, NULL)) | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_create_source (G_SOCKET (self), condition, cancellable); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GIOCondition | 
					
						
							|  |  |  |  | g_socket_datagram_based_condition_check (GDatagramBased  *datagram_based, | 
					
						
							|  |  |  |  |                                          GIOCondition     condition) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!check_datagram_based (datagram_based, NULL)) | 
					
						
							|  |  |  |  |     return G_IO_ERR; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_condition_check (G_SOCKET (datagram_based), condition); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							|  |  |  |  | g_socket_datagram_based_condition_wait (GDatagramBased  *datagram_based, | 
					
						
							|  |  |  |  |                                         GIOCondition     condition, | 
					
						
							|  |  |  |  |                                         gint64           timeout, | 
					
						
							|  |  |  |  |                                         GCancellable    *cancellable, | 
					
						
							|  |  |  |  |                                         GError         **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!check_datagram_based (datagram_based, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_condition_timed_wait (G_SOCKET (datagram_based), condition, | 
					
						
							|  |  |  |  |                                         timeout, cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_new: | 
					
						
							|  |  |  |  |  * @family: the socket family to use, e.g. %G_SOCKET_FAMILY_IPV4. | 
					
						
							|  |  |  |  |  * @type: the socket type to use. | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  |  * @protocol: the id of the protocol to use, or 0 for default. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Creates a new #GSocket with the defined family, type and protocol. | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  |  * If @protocol is 0 (%G_SOCKET_PROTOCOL_DEFAULT) the default protocol type | 
					
						
							|  |  |  |  |  * for the family and type is used. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * The @protocol is a family and type specific int that specifies what | 
					
						
							|  |  |  |  |  * kind of protocol to use. #GSocketProtocol lists several common ones. | 
					
						
							|  |  |  |  |  * Many families only support one protocol, and use 0 for this, others | 
					
						
							|  |  |  |  |  * support several and using 0 means to use the default protocol for | 
					
						
							|  |  |  |  |  * the family and type. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  |  * The protocol id is passed directly to the operating | 
					
						
							|  |  |  |  |  * system, so you can use protocols not listed in #GSocketProtocol if you | 
					
						
							|  |  |  |  |  * know the protocol number used for it. | 
					
						
							| 
									
										
										
										
											2009-05-15 09:10:23 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Returns: a #GSocket or %NULL on error. | 
					
						
							|  |  |  |  |  *     Free the returned object with g_object_unref(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSocket * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_new (GSocketFamily     family, | 
					
						
							|  |  |  |  | 	      GSocketType       type, | 
					
						
							|  |  |  |  | 	      GSocketProtocol   protocol, | 
					
						
							|  |  |  |  | 	      GError          **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   return G_SOCKET (g_initable_new (G_TYPE_SOCKET, | 
					
						
							|  |  |  |  | 				   NULL, error, | 
					
						
							|  |  |  |  | 				   "family", family, | 
					
						
							|  |  |  |  | 				   "type", type, | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  | 				   "protocol", protocol, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 				   NULL)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_new_from_fd: | 
					
						
							|  |  |  |  |  * @fd: a native socket file descriptor. | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Creates a new #GSocket from a native file descriptor | 
					
						
							|  |  |  |  |  * or winsock SOCKET handle. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This reads all the settings from the file descriptor so that | 
					
						
							|  |  |  |  |  * all properties should work. Note that the file descriptor | 
					
						
							|  |  |  |  |  * will be set to non-blocking mode, independent on the blocking | 
					
						
							|  |  |  |  |  * mode of the #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-05-15 11:44:18 +01:00
										 |  |  |  |  * On success, the returned #GSocket takes ownership of @fd. On failure, the | 
					
						
							|  |  |  |  |  * caller must close @fd themselves. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-03-17 13:06:02 +01:00
										 |  |  |  |  * Since GLib 2.46, it is no longer a fatal error to call this on a non-socket | 
					
						
							|  |  |  |  |  * descriptor.  Instead, a GError will be set with code %G_IO_ERROR_FAILED | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Returns: a #GSocket or %NULL on error. | 
					
						
							|  |  |  |  |  *     Free the returned object with g_object_unref(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSocket * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_new_from_fd (gint     fd, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		      GError **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   return G_SOCKET (g_initable_new (G_TYPE_SOCKET, | 
					
						
							|  |  |  |  | 				   NULL, error, | 
					
						
							|  |  |  |  | 				   "fd", fd, | 
					
						
							|  |  |  |  | 				   NULL)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_blocking: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @blocking: Whether to use blocking I/O or not. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sets the blocking mode of the socket. In blocking mode | 
					
						
							| 
									
										
										
										
											2015-07-29 14:08:31 +01:00
										 |  |  |  |  * all operations (which don’t take an explicit blocking parameter) block until | 
					
						
							|  |  |  |  |  * they succeed or there is an error. In | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * non-blocking mode all functions return results immediately or | 
					
						
							|  |  |  |  |  * with a %G_IO_ERROR_WOULD_BLOCK error. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * All sockets are created in blocking mode. However, note that the | 
					
						
							|  |  |  |  |  * platform level socket is always non-blocking, and blocking mode | 
					
						
							|  |  |  |  |  * is a GSocket level feature. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | void | 
					
						
							|  |  |  |  | g_socket_set_blocking (GSocket  *socket, | 
					
						
							|  |  |  |  | 		       gboolean  blocking) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   blocking = !!blocking; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->blocking == blocking) | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket->priv->blocking = blocking; | 
					
						
							|  |  |  |  |   g_object_notify (G_OBJECT (socket), "blocking"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_blocking: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the blocking mode of the socket. For details on blocking I/O, | 
					
						
							|  |  |  |  |  * see g_socket_set_blocking(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE if blocking I/O is used, %FALSE otherwise. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_get_blocking (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->blocking; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_keepalive: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * @keepalive: Value for the keepalive flag | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * Sets or unsets the %SO_KEEPALIVE flag on the underlying socket. When | 
					
						
							|  |  |  |  |  * this flag is set on a socket, the system will attempt to verify that the | 
					
						
							|  |  |  |  |  * remote socket endpoint is still present if a sufficiently long period of | 
					
						
							|  |  |  |  |  * time passes with no data being exchanged. If the system is unable to | 
					
						
							|  |  |  |  |  * verify the presence of the remote endpoint, it will automatically close | 
					
						
							|  |  |  |  |  * the connection. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * This option is only functional on certain kinds of sockets. (Notably, | 
					
						
							|  |  |  |  |  * %G_SOCKET_PROTOCOL_TCP sockets.) | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * The exact time between pings is system- and protocol-dependent, but will | 
					
						
							|  |  |  |  |  * normally be at least two hours. Most commonly, you would set this flag | 
					
						
							|  |  |  |  |  * on a server socket if you want to allow clients to remain idle for long | 
					
						
							|  |  |  |  |  * periods of time, but also want to ensure that connections are eventually | 
					
						
							|  |  |  |  |  * garbage-collected if clients crash or become unreachable. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | void | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_set_keepalive (GSocket  *socket, | 
					
						
							|  |  |  |  | 			gboolean  keepalive) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   keepalive = !!keepalive; | 
					
						
							|  |  |  |  |   if (socket->priv->keepalive == keepalive) | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (!g_socket_set_option (socket, SOL_SOCKET, SO_KEEPALIVE, | 
					
						
							|  |  |  |  | 			    keepalive, &error)) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error setting keepalive: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket->priv->keepalive = keepalive; | 
					
						
							|  |  |  |  |   g_object_notify (G_OBJECT (socket), "keepalive"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_keepalive: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the keepalive mode of the socket. For details on this, | 
					
						
							|  |  |  |  |  * see g_socket_set_keepalive(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE if keepalive is active, %FALSE otherwise. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_get_keepalive (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->keepalive; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_listen_backlog: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the listen backlog setting of the socket. For details on this, | 
					
						
							|  |  |  |  |  * see g_socket_set_listen_backlog(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the maximum number of pending connections. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gint | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_get_listen_backlog  (GSocket *socket) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), 0); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->listen_backlog; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_listen_backlog: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @backlog: the maximum number of pending connections. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sets the maximum number of outstanding connections allowed | 
					
						
							|  |  |  |  |  * when listening on this socket. If more clients than this are | 
					
						
							|  |  |  |  |  * connecting to the socket and the application is not handling them | 
					
						
							|  |  |  |  |  * on time then the new connections will be refused. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-19 10:47:55 +02:00
										 |  |  |  |  * Note that this must be called before g_socket_listen() and has no | 
					
						
							|  |  |  |  |  * effect if called after that. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | void | 
					
						
							|  |  |  |  | g_socket_set_listen_backlog (GSocket *socket, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 			     gint     backlog) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							| 
									
										
										
										
											2009-05-19 10:47:55 +02:00
										 |  |  |  |   g_return_if_fail (!socket->priv->listening); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (backlog != socket->priv->listen_backlog) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       socket->priv->listen_backlog = backlog; | 
					
						
							|  |  |  |  |       g_object_notify (G_OBJECT (socket), "listen-backlog"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_timeout: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the timeout setting of the socket. For details on this, see | 
					
						
							|  |  |  |  |  * g_socket_set_timeout(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the timeout in seconds | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.26 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | guint | 
					
						
							|  |  |  |  | g_socket_get_timeout (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), 0); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->timeout; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_timeout: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @timeout: the timeout for @socket, in seconds, or 0 for none | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sets the time in seconds after which I/O operations on @socket will | 
					
						
							|  |  |  |  |  * time out if they have not yet completed. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * On a blocking socket, this means that any blocking #GSocket | 
					
						
							|  |  |  |  |  * operation will time out after @timeout seconds of inactivity, | 
					
						
							|  |  |  |  |  * returning %G_IO_ERROR_TIMED_OUT. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * On a non-blocking socket, calls to g_socket_condition_wait() will | 
					
						
							|  |  |  |  |  * also fail with %G_IO_ERROR_TIMED_OUT after the given time. Sources | 
					
						
							|  |  |  |  |  * created with g_socket_create_source() will trigger after | 
					
						
							|  |  |  |  |  * @timeout seconds of inactivity, with the requested condition | 
					
						
							|  |  |  |  |  * set, at which point calling g_socket_receive(), g_socket_send(), | 
					
						
							|  |  |  |  |  * g_socket_check_connect_result(), etc, will fail with | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_TIMED_OUT. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If @timeout is 0 (the default), operations will never time out | 
					
						
							|  |  |  |  |  * on their own. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Note that if an I/O operation is interrupted by a signal, this may | 
					
						
							|  |  |  |  |  * cause the timeout to be reset. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.26 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | g_socket_set_timeout (GSocket *socket, | 
					
						
							|  |  |  |  | 		      guint    timeout) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (timeout != socket->priv->timeout) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       socket->priv->timeout = timeout; | 
					
						
							|  |  |  |  |       g_object_notify (G_OBJECT (socket), "timeout"); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_ttl: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the unicast time-to-live setting on @socket; see | 
					
						
							|  |  |  |  |  * g_socket_set_ttl() for more details. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the time-to-live setting on @socket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | guint | 
					
						
							|  |  |  |  | g_socket_get_ttl (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |  |   gint value; | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), 0); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->family == G_SOCKET_FAMILY_IPV4) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_get_option (socket, IPPROTO_IP, IP_TTL, | 
					
						
							|  |  |  |  | 			   &value, &error); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else if (socket->priv->family == G_SOCKET_FAMILY_IPV6) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_get_option (socket, IPPROTO_IPV6, IPV6_UNICAST_HOPS, | 
					
						
							|  |  |  |  | 			   &value, &error); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |     g_return_val_if_reached (0); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (error) | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error getting unicast ttl: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							|  |  |  |  |       return 0; | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return value; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_ttl: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @ttl: the time-to-live value for all unicast packets on @socket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sets the time-to-live for outgoing unicast packets on @socket. | 
					
						
							|  |  |  |  |  * By default the platform-specific default value is used. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | g_socket_set_ttl (GSocket  *socket, | 
					
						
							|  |  |  |  |                   guint     ttl) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->family == G_SOCKET_FAMILY_IPV4) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IP, IP_TTL, | 
					
						
							|  |  |  |  | 			   ttl, &error); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else if (socket->priv->family == G_SOCKET_FAMILY_IPV6) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-12-12 16:39:02 +01:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IP, IP_TTL, | 
					
						
							|  |  |  |  | 			   ttl, NULL); | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IPV6, IPV6_UNICAST_HOPS, | 
					
						
							|  |  |  |  | 			   ttl, &error); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     g_return_if_reached (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (error) | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error setting unicast ttl: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:48:02 +01:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_notify (G_OBJECT (socket), "ttl"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_broadcast: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the broadcast setting on @socket; if %TRUE, | 
					
						
							|  |  |  |  |  * it is possible to send packets to broadcast | 
					
						
							| 
									
										
										
										
											2013-02-17 09:41:55 -05:00
										 |  |  |  |  * addresses. | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the broadcast setting on @socket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_get_broadcast (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |  |   gint value; | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (!g_socket_get_option (socket, SOL_SOCKET, SO_BROADCAST, | 
					
						
							|  |  |  |  | 			    &value, &error)) | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error getting broadcast: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return !!value; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_broadcast: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							| 
									
										
										
										
											2013-02-17 09:41:55 -05:00
										 |  |  |  |  * @broadcast: whether @socket should allow sending to broadcast | 
					
						
							|  |  |  |  |  *     addresses | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-02-17 09:41:55 -05:00
										 |  |  |  |  * Sets whether @socket should allow sending to broadcast addresses. | 
					
						
							|  |  |  |  |  * This is %FALSE by default. | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | g_socket_set_broadcast (GSocket    *socket, | 
					
						
							|  |  |  |  |        	                gboolean    broadcast) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   broadcast = !!broadcast; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (!g_socket_set_option (socket, SOL_SOCKET, SO_BROADCAST, | 
					
						
							|  |  |  |  | 			    broadcast, &error)) | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error setting broadcast: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2012-01-13 13:01:35 +01:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_notify (G_OBJECT (socket), "broadcast"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_multicast_loopback: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the multicast loopback setting on @socket; if %TRUE (the | 
					
						
							|  |  |  |  |  * default), outgoing multicast packets will be looped back to | 
					
						
							|  |  |  |  |  * multicast listeners on the same host. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the multicast loopback setting on @socket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_get_multicast_loopback (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |  |   gint value; | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->family == G_SOCKET_FAMILY_IPV4) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_get_option (socket, IPPROTO_IP, IP_MULTICAST_LOOP, | 
					
						
							|  |  |  |  | 			   &value, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else if (socket->priv->family == G_SOCKET_FAMILY_IPV6) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_get_option (socket, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, | 
					
						
							|  |  |  |  | 			   &value, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     g_return_val_if_reached (FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (error) | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error getting multicast loopback: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return !!value; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_multicast_loopback: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @loopback: whether @socket should receive messages sent to its | 
					
						
							|  |  |  |  |  *   multicast groups from the local host | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sets whether outgoing multicast packets will be received by sockets | 
					
						
							|  |  |  |  |  * listening on that multicast address on the same host. This is %TRUE | 
					
						
							|  |  |  |  |  * by default. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | g_socket_set_multicast_loopback (GSocket    *socket, | 
					
						
							|  |  |  |  | 				 gboolean    loopback) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   loopback = !!loopback; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->family == G_SOCKET_FAMILY_IPV4) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IP, IP_MULTICAST_LOOP, | 
					
						
							|  |  |  |  | 			   loopback, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else if (socket->priv->family == G_SOCKET_FAMILY_IPV6) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-12-12 16:39:02 +01:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IP, IP_MULTICAST_LOOP, | 
					
						
							|  |  |  |  | 			   loopback, NULL); | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, | 
					
						
							|  |  |  |  | 			   loopback, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     g_return_if_reached (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (error) | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error setting multicast loopback: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_notify (G_OBJECT (socket), "multicast-loopback"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_multicast_ttl: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the multicast time-to-live setting on @socket; see | 
					
						
							|  |  |  |  |  * g_socket_set_multicast_ttl() for more details. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the multicast time-to-live setting on @socket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | guint | 
					
						
							|  |  |  |  | g_socket_get_multicast_ttl (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							|  |  |  |  |   gint value; | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 16:39:02 +01:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), 0); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->family == G_SOCKET_FAMILY_IPV4) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_get_option (socket, IPPROTO_IP, IP_MULTICAST_TTL, | 
					
						
							|  |  |  |  | 			   &value, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else if (socket->priv->family == G_SOCKET_FAMILY_IPV6) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_get_option (socket, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, | 
					
						
							|  |  |  |  | 			   &value, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     g_return_val_if_reached (FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (error) | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error getting multicast ttl: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return value; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_multicast_ttl: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @ttl: the time-to-live value for all multicast datagrams on @socket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sets the time-to-live for outgoing multicast datagrams on @socket. | 
					
						
							|  |  |  |  |  * By default, this is 1, meaning that multicast packets will not leave | 
					
						
							|  |  |  |  |  * the local network. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | g_socket_set_multicast_ttl (GSocket  *socket, | 
					
						
							|  |  |  |  |                             guint     ttl) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   GError *error = NULL; | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_if_fail (G_IS_SOCKET (socket)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->family == G_SOCKET_FAMILY_IPV4) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IP, IP_MULTICAST_TTL, | 
					
						
							|  |  |  |  | 			   ttl, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else if (socket->priv->family == G_SOCKET_FAMILY_IPV6) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-12-12 16:39:02 +01:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IP, IP_MULTICAST_TTL, | 
					
						
							|  |  |  |  | 			   ttl, NULL); | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_socket_set_option (socket, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, | 
					
						
							|  |  |  |  | 			   ttl, &error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     g_return_if_reached (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (error) | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_warning ("error setting multicast ttl: %s", error->message); | 
					
						
							|  |  |  |  |       g_error_free (error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_notify (G_OBJECT (socket), "multicast-ttl"); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_family: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the socket family of the socket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: a #GSocketFamily | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSocketFamily | 
					
						
							|  |  |  |  | g_socket_get_family (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), G_SOCKET_FAMILY_INVALID); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->family; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_socket_type: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the socket type of the socket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: a #GSocketType | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSocketType | 
					
						
							|  |  |  |  | g_socket_get_socket_type (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), G_SOCKET_TYPE_INVALID); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->type; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  |  * g_socket_get_protocol: | 
					
						
							| 
									
										
										
										
											2009-05-15 09:10:23 +02:00
										 |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the socket protocol id the socket was created with. | 
					
						
							|  |  |  |  |  * In case the protocol is unknown, -1 is returned. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: a protocol id, or -1 if unknown | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-20 12:01:29 +02:00
										 |  |  |  | GSocketProtocol | 
					
						
							|  |  |  |  | g_socket_get_protocol (GSocket *socket) | 
					
						
							| 
									
										
										
										
											2009-05-15 09:10:23 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->protocol; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_fd: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns the underlying OS socket object. On unix this | 
					
						
							| 
									
										
										
										
											2012-08-16 23:02:41 -04:00
										 |  |  |  |  * is a socket file descriptor, and on Windows this is | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * a Winsock2 SOCKET handle. This may be useful for | 
					
						
							|  |  |  |  |  * doing platform specific or otherwise unusual operations | 
					
						
							|  |  |  |  |  * on the socket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the file descriptor of the socket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | int | 
					
						
							|  |  |  |  | g_socket_get_fd (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return socket->priv->fd; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_local_address: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Try to get the local address of a bound socket. This is only | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |  * useful if the socket has been bound to a local address, | 
					
						
							|  |  |  |  |  * either explicitly or implicitly when connecting. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-09-24 18:24:41 -03:00
										 |  |  |  |  * Returns: (transfer full): a #GSocketAddress or %NULL on error. | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |  *     Free the returned object with g_object_unref(). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSocketAddress * | 
					
						
							|  |  |  |  | g_socket_get_local_address (GSocket  *socket, | 
					
						
							|  |  |  |  | 			    GError  **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   struct sockaddr_storage buffer; | 
					
						
							| 
									
										
										
										
											2012-11-11 13:05:16 -05:00
										 |  |  |  |   guint len = sizeof (buffer); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   if (getsockname (socket->priv->fd, (struct sockaddr *) &buffer, &len) < 0) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 		   _("could not get local address: %s"), socket_strerror (errsv)); | 
					
						
							|  |  |  |  |       return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   return g_socket_address_new_from_native (&buffer, len); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_remote_address: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Try to get the remove address of a connected socket. This is only | 
					
						
							|  |  |  |  |  * useful for connection oriented sockets that have been connected. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-09-24 18:24:41 -03:00
										 |  |  |  |  * Returns: (transfer full): a #GSocketAddress or %NULL on error. | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |  *     Free the returned object with g_object_unref(). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSocketAddress * | 
					
						
							|  |  |  |  | g_socket_get_remote_address (GSocket  *socket, | 
					
						
							|  |  |  |  | 			     GError  **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   struct sockaddr_storage buffer; | 
					
						
							| 
									
										
										
										
											2012-11-11 13:05:16 -05:00
										 |  |  |  |   guint len = sizeof (buffer); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |   if (socket->priv->connect_pending) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |       if (!g_socket_check_connect_result (socket, error)) | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         socket->priv->connect_pending = FALSE; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |   if (!socket->priv->remote_address) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (getpeername (socket->priv->fd, (struct sockaddr *) &buffer, &len) < 0) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 	  g_set_error (error, G_IO_ERROR, socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 		       _("could not get remote address: %s"), socket_strerror (errsv)); | 
					
						
							|  |  |  |  | 	  return NULL; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       socket->priv->remote_address = g_socket_address_new_from_native (&buffer, len); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_object_ref (socket->priv->remote_address); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_is_connected: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Check whether the socket is connected. This is only useful for | 
					
						
							|  |  |  |  |  * connection-oriented sockets. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |  * If using g_socket_shutdown(), this function will return %TRUE until the | 
					
						
							|  |  |  |  |  * socket has been shut down for reading and writing. If you do a non-blocking | 
					
						
							|  |  |  |  |  * connect, this function will not return %TRUE until after you call | 
					
						
							|  |  |  |  |  * g_socket_check_connect_result(). | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Returns: %TRUE if socket is connected, %FALSE otherwise. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_is_connected (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |   return (socket->priv->connected_read || socket->priv->connected_write); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_listen: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Marks the socket as a server socket, i.e. a socket that is used | 
					
						
							|  |  |  |  |  * to accept incoming requests using g_socket_accept(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Before calling this the socket must be bound to a local address using | 
					
						
							|  |  |  |  |  * g_socket_bind(). | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-19 10:47:55 +02:00
										 |  |  |  |  * To set the maximum amount of outstanding clients, use | 
					
						
							|  |  |  |  |  * g_socket_set_listen_backlog(). | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Returns: %TRUE on success, %FALSE on error. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_listen (GSocket  *socket, | 
					
						
							|  |  |  |  | 		 GError  **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (listen (socket->priv->fd, socket->priv->listen_backlog) < 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 		   _("could not listen: %s"), socket_strerror (errsv)); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:47:55 +02:00
										 |  |  |  |   socket->priv->listening = TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_bind: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @address: a #GSocketAddress specifying the local address. | 
					
						
							|  |  |  |  |  * @allow_reuse: whether to allow reusing this address | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * When a socket is created it is attached to an address family, but it | 
					
						
							|  |  |  |  |  * doesn't have an address in this family. g_socket_bind() assigns the | 
					
						
							|  |  |  |  |  * address (sometimes called name) of the socket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * It is generally required to bind to a local address before you can | 
					
						
							|  |  |  |  |  * receive connections. (See g_socket_listen() and g_socket_accept() ). | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * In certain situations, you may also want to bind a socket that will be | 
					
						
							|  |  |  |  |  * used to initiate connections, though this is not normally required. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-02-17 15:11:18 -05:00
										 |  |  |  |  * If @socket is a TCP socket, then @allow_reuse controls the setting | 
					
						
							| 
									
										
										
										
											2014-02-06 08:04:52 -05:00
										 |  |  |  |  * of the `SO_REUSEADDR` socket option; normally it should be %TRUE for | 
					
						
							|  |  |  |  |  * server sockets (sockets that you will eventually call | 
					
						
							|  |  |  |  |  * g_socket_accept() on), and %FALSE for client sockets. (Failing to | 
					
						
							|  |  |  |  |  * set this flag on a server socket may cause g_socket_bind() to return | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_ADDRESS_IN_USE if the server program is stopped and then | 
					
						
							|  |  |  |  |  * immediately restarted.) | 
					
						
							| 
									
										
										
										
											2013-02-17 15:11:18 -05:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * If @socket is a UDP socket, then @allow_reuse determines whether or | 
					
						
							|  |  |  |  |  * not other UDP sockets can be bound to the same address at the same | 
					
						
							|  |  |  |  |  * time. In particular, you can have several UDP sockets bound to the | 
					
						
							|  |  |  |  |  * same address, and they will all receive all of the multicast and | 
					
						
							|  |  |  |  |  * broadcast packets sent to that address. (The behavior of unicast | 
					
						
							|  |  |  |  |  * UDP packets to an address with multiple listeners is not defined.) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE on success, %FALSE on error. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_bind (GSocket         *socket, | 
					
						
							|  |  |  |  | 	       GSocketAddress  *address, | 
					
						
							|  |  |  |  | 	       gboolean         reuse_address, | 
					
						
							|  |  |  |  | 	       GError         **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   struct sockaddr_storage addr; | 
					
						
							| 
									
										
										
										
											2013-02-17 15:11:18 -05:00
										 |  |  |  |   gboolean so_reuseaddr; | 
					
						
							|  |  |  |  | #ifdef SO_REUSEPORT
 | 
					
						
							|  |  |  |  |   gboolean so_reuseport; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket) && G_IS_SOCKET_ADDRESS (address), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   if (!g_socket_address_to_native (address, &addr, sizeof addr, error)) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-17 15:11:18 -05:00
										 |  |  |  |   /* On Windows, SO_REUSEADDR has the semantics we want for UDP
 | 
					
						
							|  |  |  |  |    * sockets, but has nasty side effects we don't want for TCP | 
					
						
							|  |  |  |  |    * sockets. | 
					
						
							|  |  |  |  |    * | 
					
						
							|  |  |  |  |    * On other platforms, we set SO_REUSEPORT, if it exists, for | 
					
						
							|  |  |  |  |    * UDP sockets, and SO_REUSEADDR for all sockets, hoping that | 
					
						
							|  |  |  |  |    * if SO_REUSEPORT doesn't exist, then SO_REUSEADDR will have | 
					
						
							|  |  |  |  |    * the desired semantics on UDP (as it does on Linux, although | 
					
						
							|  |  |  |  |    * Linux has SO_REUSEPORT too as of 3.9). | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   so_reuseaddr = reuse_address && (socket->priv->type == G_SOCKET_TYPE_DATAGRAM); | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   so_reuseaddr = !!reuse_address; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef SO_REUSEPORT
 | 
					
						
							|  |  |  |  |   so_reuseport = reuse_address && (socket->priv->type == G_SOCKET_TYPE_DATAGRAM); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* Ignore errors here, the only likely error is "not supported", and
 | 
					
						
							|  |  |  |  |    * this is a "best effort" thing mainly. | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   g_socket_set_option (socket, SOL_SOCKET, SO_REUSEADDR, so_reuseaddr, NULL); | 
					
						
							|  |  |  |  | #ifdef SO_REUSEPORT
 | 
					
						
							|  |  |  |  |   g_socket_set_option (socket, SOL_SOCKET, SO_REUSEPORT, so_reuseport, NULL); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   if (bind (socket->priv->fd, (struct sockaddr *) &addr, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	    g_socket_address_get_native_size (address)) < 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  |       g_set_error (error, | 
					
						
							|  |  |  |  | 		   G_IO_ERROR, socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 		   _("Error binding to address: %s"), socket_strerror (errsv)); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-31 14:11:55 +02:00
										 |  |  |  | #if !defined(HAVE_IF_NAMETOINDEX) && defined(G_OS_WIN32)
 | 
					
						
							|  |  |  |  | static guint | 
					
						
							|  |  |  |  | if_nametoindex (const gchar *iface) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   PIP_ADAPTER_ADDRESSES addresses = NULL, p; | 
					
						
							|  |  |  |  |   gulong addresses_len = 0; | 
					
						
							|  |  |  |  |   guint idx = 0; | 
					
						
							|  |  |  |  |   DWORD res; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-05 16:13:03 +08:00
										 |  |  |  |   if (ws2funcs.pIfNameToIndex != NULL) | 
					
						
							|  |  |  |  |     return ws2funcs.pIfNameToIndex (iface); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-31 14:11:55 +02:00
										 |  |  |  |   res = GetAdaptersAddresses (AF_UNSPEC, 0, NULL, NULL, &addresses_len); | 
					
						
							|  |  |  |  |   if (res != NO_ERROR && res != ERROR_BUFFER_OVERFLOW) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (res == ERROR_NO_DATA) | 
					
						
							|  |  |  |  |         errno = ENXIO; | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         errno = EINVAL; | 
					
						
							|  |  |  |  |       return 0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   addresses = g_malloc (addresses_len); | 
					
						
							|  |  |  |  |   res = GetAdaptersAddresses (AF_UNSPEC, 0, NULL, addresses, &addresses_len); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (res != NO_ERROR) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_free (addresses); | 
					
						
							|  |  |  |  |       if (res == ERROR_NO_DATA) | 
					
						
							|  |  |  |  |         errno = ENXIO; | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         errno = EINVAL; | 
					
						
							|  |  |  |  |       return 0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   p = addresses; | 
					
						
							|  |  |  |  |   while (p) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (strcmp (p->AdapterName, iface) == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           idx = p->IfIndex; | 
					
						
							|  |  |  |  |           break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       p = p->Next; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (p == NULL) | 
					
						
							|  |  |  |  |     errno = ENXIO; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_free (addresses); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return idx; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #define HAVE_IF_NAMETOINDEX 1
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | static gboolean | 
					
						
							|  |  |  |  | g_socket_multicast_group_operation (GSocket       *socket, | 
					
						
							|  |  |  |  | 				    GInetAddress  *group, | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |                                     gboolean       source_specific, | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  |                                     const gchar   *iface, | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 				    gboolean       join_group, | 
					
						
							|  |  |  |  | 				    GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   const guint8 *native_addr; | 
					
						
							|  |  |  |  |   gint optname, result; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (socket->priv->type == G_SOCKET_TYPE_DATAGRAM, FALSE); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_INET_ADDRESS (group), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   native_addr = g_inet_address_to_bytes (group); | 
					
						
							| 
									
										
										
										
											2012-12-12 16:39:02 +01:00
										 |  |  |  |   if (g_inet_address_get_family (group) == G_SOCKET_FAMILY_IPV4) | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  | #ifdef HAVE_IP_MREQN
 | 
					
						
							|  |  |  |  |       struct ip_mreqn mc_req; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       struct ip_mreq mc_req; | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-15 11:37:20 +01:00
										 |  |  |  |       memset (&mc_req, 0, sizeof (mc_req)); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       memcpy (&mc_req.imr_multiaddr, native_addr, sizeof (struct in_addr)); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef HAVE_IP_MREQN
 | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  |       if (iface) | 
					
						
							|  |  |  |  |         mc_req.imr_ifindex = if_nametoindex (iface); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  |       else | 
					
						
							|  |  |  |  |         mc_req.imr_ifindex = 0;  /* Pick any.  */ | 
					
						
							| 
									
										
										
										
											2013-07-31 14:11:55 +02:00
										 |  |  |  | #elif defined(G_OS_WIN32)
 | 
					
						
							|  |  |  |  |       if (iface) | 
					
						
							|  |  |  |  |         mc_req.imr_interface.s_addr = g_htonl (if_nametoindex (iface)); | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         mc_req.imr_interface.s_addr = g_htonl (INADDR_ANY); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       mc_req.imr_interface.s_addr = g_htonl (INADDR_ANY); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |       if (source_specific) | 
					
						
							| 
									
										
										
										
											2012-01-24 08:28:33 -05:00
										 |  |  |  | 	{ | 
					
						
							|  |  |  |  | #ifdef IP_ADD_SOURCE_MEMBERSHIP
 | 
					
						
							|  |  |  |  | 	  optname = join_group ? IP_ADD_SOURCE_MEMBERSHIP : IP_DROP_SOURCE_MEMBERSHIP; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  | 	  g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, | 
					
						
							|  |  |  |  | 		       join_group ? | 
					
						
							|  |  |  |  | 		       _("Error joining multicast group: %s") : | 
					
						
							|  |  |  |  | 		       _("Error leaving multicast group: %s"), | 
					
						
							|  |  |  |  | 		       _("No support for source-specific multicast")); | 
					
						
							|  |  |  |  | 	  return FALSE; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |       else | 
					
						
							|  |  |  |  |         optname = join_group ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP; | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       result = setsockopt (socket->priv->fd, IPPROTO_IP, optname, | 
					
						
							|  |  |  |  | 			   &mc_req, sizeof (mc_req)); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-12-12 16:39:02 +01:00
										 |  |  |  |   else if (g_inet_address_get_family (group) == G_SOCKET_FAMILY_IPV6) | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |       struct ipv6_mreq mc_req_ipv6; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-15 11:37:20 +01:00
										 |  |  |  |       memset (&mc_req_ipv6, 0, sizeof (mc_req_ipv6)); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       memcpy (&mc_req_ipv6.ipv6mr_multiaddr, native_addr, sizeof (struct in6_addr)); | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  | #ifdef HAVE_IF_NAMETOINDEX
 | 
					
						
							|  |  |  |  |       if (iface) | 
					
						
							|  |  |  |  |         mc_req_ipv6.ipv6mr_interface = if_nametoindex (iface); | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  |       else | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-01-13 12:37:31 +01:00
										 |  |  |  |         mc_req_ipv6.ipv6mr_interface = 0; | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-16 17:15:06 -05:00
										 |  |  |  |       optname = join_group ? IPV6_JOIN_GROUP : IPV6_LEAVE_GROUP; | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |       result = setsockopt (socket->priv->fd, IPPROTO_IPV6, optname, | 
					
						
							|  |  |  |  | 			   &mc_req_ipv6, sizeof (mc_req_ipv6)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     g_return_val_if_reached (FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (result < 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 		   join_group ? | 
					
						
							|  |  |  |  | 		   _("Error joining multicast group: %s") : | 
					
						
							|  |  |  |  | 		   _("Error leaving multicast group: %s"), | 
					
						
							|  |  |  |  | 		   socket_strerror (errsv)); | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_join_multicast_group: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @group: a #GInetAddress specifying the group address to join. | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @iface: (nullable): Name of the interface to use, or %NULL | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |  * @source_specific: %TRUE if source-specific multicast should be used | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Registers @socket to receive multicast messages sent to @group. | 
					
						
							|  |  |  |  |  * @socket must be a %G_SOCKET_TYPE_DATAGRAM socket, and must have | 
					
						
							|  |  |  |  |  * been bound to an appropriate interface and port with | 
					
						
							|  |  |  |  |  * g_socket_bind(). | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-01-24 08:28:33 -05:00
										 |  |  |  |  * If @iface is %NULL, the system will automatically pick an interface | 
					
						
							|  |  |  |  |  * to bind to based on @group. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |  * If @source_specific is %TRUE, source-specific multicast as defined | 
					
						
							| 
									
										
										
										
											2012-01-24 08:28:33 -05:00
										 |  |  |  |  * in RFC 4604 is used. Note that on older platforms this may fail | 
					
						
							|  |  |  |  |  * with a %G_IO_ERROR_NOT_SUPPORTED error. | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |  * Returns: %TRUE on success, %FALSE on error. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_join_multicast_group (GSocket       *socket, | 
					
						
							|  |  |  |  | 			       GInetAddress  *group, | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |                                gboolean       source_specific, | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  |                                const gchar   *iface, | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 			       GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  |   return g_socket_multicast_group_operation (socket, group, source_specific, iface, TRUE, error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_leave_multicast_group: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @group: a #GInetAddress specifying the group address to leave. | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @iface: (nullable): Interface used | 
					
						
							| 
									
										
										
										
											2012-01-24 08:28:33 -05:00
										 |  |  |  |  * @source_specific: %TRUE if source-specific multicast was used | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-01-24 08:28:33 -05:00
										 |  |  |  |  * Removes @socket from the multicast group defined by @group, @iface, | 
					
						
							|  |  |  |  |  * and @source_specific (which must all have the same values they had | 
					
						
							|  |  |  |  |  * when you joined the group). | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-01-24 08:28:33 -05:00
										 |  |  |  |  * @socket remains bound to its address and port, and can still receive | 
					
						
							|  |  |  |  |  * unicast messages after calling this. | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  |  * Returns: %TRUE on success, %FALSE on error. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_leave_multicast_group (GSocket       *socket, | 
					
						
							|  |  |  |  | 				GInetAddress  *group, | 
					
						
							| 
									
										
										
										
											2012-01-13 12:53:50 +01:00
										 |  |  |  |                                 gboolean       source_specific, | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  |                                 const gchar   *iface, | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | 				GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-01-17 19:46:34 -05:00
										 |  |  |  |   return g_socket_multicast_group_operation (socket, group, source_specific, iface, FALSE, error); | 
					
						
							| 
									
										
										
										
											2011-12-29 11:01:23 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-12 12:21:07 -04:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_speaks_ipv4: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Checks if a socket is capable of speaking IPv4. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * IPv4 sockets are capable of speaking IPv4.  On some operating systems | 
					
						
							|  |  |  |  |  * and under some combinations of circumstances IPv6 sockets are also | 
					
						
							|  |  |  |  |  * capable of speaking IPv4.  See RFC 3493 section 3.7 for more | 
					
						
							|  |  |  |  |  * information. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * No other types of sockets are currently considered as being capable | 
					
						
							|  |  |  |  |  * of speaking IPv4. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE if this socket can be used with IPv4. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-01-30 14:06:12 -05:00
										 |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-06-12 12:21:07 -04:00
										 |  |  |  |  **/ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_speaks_ipv4 (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   switch (socket->priv->family) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |     case G_SOCKET_FAMILY_IPV4: | 
					
						
							|  |  |  |  |       return TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     case G_SOCKET_FAMILY_IPV6: | 
					
						
							|  |  |  |  | #if defined (IPPROTO_IPV6) && defined (IPV6_V6ONLY)
 | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         gint v6_only; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |         if (!g_socket_get_option (socket, | 
					
						
							|  |  |  |  | 				  IPPROTO_IPV6, IPV6_V6ONLY, | 
					
						
							|  |  |  |  | 				  &v6_only, NULL)) | 
					
						
							| 
									
										
										
										
											2009-06-12 12:21:07 -04:00
										 |  |  |  |           return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         return !v6_only; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     default: | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_accept: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Accept incoming connections on a connection-based socket. This removes | 
					
						
							|  |  |  |  |  * the first outstanding connection request from the listening socket and | 
					
						
							|  |  |  |  |  * creates a #GSocket object for it. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * The @socket must be bound to a local address with g_socket_bind() and | 
					
						
							|  |  |  |  |  * must be listening for incoming connections (g_socket_listen()). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If there are no outstanding connections then the operation will block | 
					
						
							|  |  |  |  |  * or return %G_IO_ERROR_WOULD_BLOCK if non-blocking I/O is enabled. | 
					
						
							|  |  |  |  |  * To be notified of an incoming connection, wait for the %G_IO_IN condition. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-09-24 18:24:41 -03:00
										 |  |  |  |  * Returns: (transfer full): a new #GSocket, or %NULL on error. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  *     Free the returned object with g_object_unref(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSocket * | 
					
						
							|  |  |  |  | g_socket_accept (GSocket       *socket, | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 		 GCancellable  *cancellable, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		 GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocket *new_socket; | 
					
						
							|  |  |  |  |   gint ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-14 17:46:38 -04:00
										 |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |   while (TRUE) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |       win32_unset_event_mask (socket, FD_ACCEPT); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       if ((ret = accept (socket->priv->fd, NULL, 0)) < 0) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  if (errsv == EINTR) | 
					
						
							|  |  |  |  | 	    continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef WSAEWOULDBLOCK
 | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |           if (errsv == WSAEWOULDBLOCK) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |           if (errsv == EWOULDBLOCK || | 
					
						
							|  |  |  |  |               errsv == EAGAIN) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |             { | 
					
						
							|  |  |  |  |               if (socket->priv->blocking) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   if (!g_socket_condition_wait (socket, | 
					
						
							|  |  |  |  |                                                 G_IO_IN, cancellable, error)) | 
					
						
							|  |  |  |  |                     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   continue; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 10:18:06 -04:00
										 |  |  |  | 	  socket_set_error_lazy (error, errsv, _("Error accepting connection: %s")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	  return NULL; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     /* The socket inherits the accepting sockets event mask and even object,
 | 
					
						
							|  |  |  |  |        we need to remove that */ | 
					
						
							|  |  |  |  |     WSAEventSelect (ret, NULL, 0); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     int flags; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* We always want to set close-on-exec to protect users. If you
 | 
					
						
							|  |  |  |  |        need to so some weird inheritance to exec you can re-enable this | 
					
						
							|  |  |  |  |        using lower level hacks with g_socket_get_fd(). */ | 
					
						
							|  |  |  |  |     flags = fcntl (ret, F_GETFD, 0); | 
					
						
							|  |  |  |  |     if (flags != -1 && | 
					
						
							|  |  |  |  | 	(flags & FD_CLOEXEC) == 0) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	flags |= FD_CLOEXEC; | 
					
						
							|  |  |  |  | 	fcntl (ret, F_SETFD, flags); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   new_socket = g_socket_new_from_fd (ret, error); | 
					
						
							|  |  |  |  |   if (new_socket == NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |       closesocket (ret); | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |       close (ret); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-15 09:10:23 +02:00
										 |  |  |  |   else | 
					
						
							|  |  |  |  |     new_socket->priv->protocol = socket->priv->protocol; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return new_socket; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_connect: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @address: a #GSocketAddress specifying the remote address. | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Connect the socket to the specified remote address. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * For connection oriented socket this generally means we attempt to make | 
					
						
							|  |  |  |  |  * a connection to the @address. For a connection-less socket it sets | 
					
						
							|  |  |  |  |  * the default address for g_socket_send() and discards all incoming datagrams | 
					
						
							|  |  |  |  |  * from other sources. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  * Generally connection oriented sockets can only connect once, but | 
					
						
							|  |  |  |  |  * connection-less sockets can connect multiple times to change the | 
					
						
							|  |  |  |  |  * default address. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * If the connect call needs to do network I/O it will block, unless | 
					
						
							|  |  |  |  |  * non-blocking I/O is enabled. Then %G_IO_ERROR_PENDING is returned | 
					
						
							|  |  |  |  |  * and the user can be notified of the connection finishing by waiting | 
					
						
							| 
									
										
										
										
											2011-07-24 14:52:03 -04:00
										 |  |  |  |  * for the G_IO_OUT condition. The result of the connection must then be | 
					
						
							| 
									
										
										
										
											2009-05-19 10:57:58 +02:00
										 |  |  |  |  * checked with g_socket_check_connect_result(). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE if connected, %FALSE on error. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_connect (GSocket         *socket, | 
					
						
							|  |  |  |  | 		  GSocketAddress  *address, | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 		  GCancellable    *cancellable, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		  GError         **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   struct sockaddr_storage buffer; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket) && G_IS_SOCKET_ADDRESS (address), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |   if (!g_socket_address_to_native (address, &buffer, sizeof buffer, error)) | 
					
						
							| 
									
										
										
										
											2009-05-18 14:10:07 +02:00
										 |  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |   if (socket->priv->remote_address) | 
					
						
							|  |  |  |  |     g_object_unref (socket->priv->remote_address); | 
					
						
							|  |  |  |  |   socket->priv->remote_address = g_object_ref (address); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   while (1) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |       win32_unset_event_mask (socket, FD_CONNECT); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:40:27 +02:00
										 |  |  |  |       if (connect (socket->priv->fd, (struct sockaddr *) &buffer, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		   g_socket_address_get_native_size (address)) < 0) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  if (errsv == EINTR) | 
					
						
							|  |  |  |  | 	    continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  | 	  if (errsv == EINPROGRESS) | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2009-05-20 16:28:20 +02:00
										 |  |  |  | 	  if (errsv == WSAEWOULDBLOCK) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 	    { | 
					
						
							|  |  |  |  | 	      if (socket->priv->blocking) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 		  if (g_socket_condition_wait (socket, G_IO_OUT, cancellable, error)) | 
					
						
							|  |  |  |  | 		    { | 
					
						
							|  |  |  |  | 		      if (g_socket_check_connect_result (socket, error)) | 
					
						
							|  |  |  |  | 			break; | 
					
						
							|  |  |  |  | 		    } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 	      else | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |                 { | 
					
						
							|  |  |  |  |                   g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PENDING, | 
					
						
							|  |  |  |  |                                        _("Connection in progress")); | 
					
						
							|  |  |  |  |                   socket->priv->connect_pending = TRUE; | 
					
						
							|  |  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	    } | 
					
						
							|  |  |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2011-11-14 19:46:02 -05:00
										 |  |  |  | 	    g_set_error_literal (error, G_IO_ERROR, | 
					
						
							|  |  |  |  | 				 socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 				 socket_strerror (errsv)); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  return FALSE; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |   socket->priv->connected_read = TRUE; | 
					
						
							|  |  |  |  |   socket->priv->connected_write = TRUE; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2009-05-19 10:57:58 +02:00
										 |  |  |  |  * g_socket_check_connect_result: | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  * Checks and resets the pending connect error for the socket. | 
					
						
							|  |  |  |  |  * This is used to check for errors when g_socket_connect() is | 
					
						
							|  |  |  |  |  * used in non-blocking mode. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE if no error, %FALSE otherwise, setting @error to the error | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							| 
									
										
										
										
											2009-05-19 10:57:58 +02:00
										 |  |  |  | g_socket_check_connect_result (GSocket  *socket, | 
					
						
							|  |  |  |  | 			       GError  **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   int value; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-04 00:13:50 +02:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-14 17:46:38 -04:00
										 |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |   if (!g_socket_get_option (socket, SOL_SOCKET, SO_ERROR, &value, error)) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |       g_prefix_error (error, _("Unable to get pending error: ")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (value != 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-05-19 15:03:14 +02:00
										 |  |  |  |       g_set_error_literal (error, G_IO_ERROR, socket_io_error_from_errno (value), | 
					
						
							|  |  |  |  |                            socket_strerror (value)); | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |       if (socket->priv->remote_address) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_object_unref (socket->priv->remote_address); | 
					
						
							|  |  |  |  |           socket->priv->remote_address = NULL; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-07-24 14:52:03 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |   socket->priv->connected_read = TRUE; | 
					
						
							|  |  |  |  |   socket->priv->connected_write = TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_available_bytes: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-10-04 11:34:36 +01:00
										 |  |  |  |  * Get the amount of data pending in the OS input buffer, without blocking. | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-08-24 13:55:06 -04:00
										 |  |  |  |  * If @socket is a UDP or SCTP socket, this will return the size of | 
					
						
							|  |  |  |  |  * just the next packet, even if additional packets are buffered after | 
					
						
							|  |  |  |  |  * that one. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Note that on Windows, this function is rather inefficient in the | 
					
						
							|  |  |  |  |  * UDP case, and so if you know any plausible upper bound on the size | 
					
						
							|  |  |  |  |  * of the incoming packet, it is better to just do a | 
					
						
							|  |  |  |  |  * g_socket_receive() with a buffer of that size, rather than calling | 
					
						
							|  |  |  |  |  * g_socket_get_available_bytes() first and then doing a receive of | 
					
						
							|  |  |  |  |  * exactly the right size. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  |  * Returns: the number of bytes that can be read from the socket | 
					
						
							| 
									
										
										
										
											2013-08-24 13:55:06 -04:00
										 |  |  |  |  * without blocking or truncating, or -1 on error. | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gssize | 
					
						
							|  |  |  |  | g_socket_get_available_bytes (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-24 13:55:06 -04:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   const gint bufsize = 64 * 1024; | 
					
						
							|  |  |  |  |   static guchar *buf = NULL; | 
					
						
							| 
									
										
										
										
											2014-02-01 14:21:10 +01:00
										 |  |  |  |   u_long avail; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2013-08-24 13:55:06 -04:00
										 |  |  |  |   gint avail; | 
					
						
							| 
									
										
										
										
											2014-02-01 14:21:10 +01:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-24 13:55:06 -04:00
										 |  |  |  | #if defined (SO_NREAD)
 | 
					
						
							|  |  |  |  |   if (!g_socket_get_option (socket, SOL_SOCKET, SO_NREAD, &avail, NULL)) | 
					
						
							|  |  |  |  |       return -1; | 
					
						
							|  |  |  |  | #elif !defined (G_OS_WIN32)
 | 
					
						
							| 
									
										
										
										
											2012-11-11 14:14:04 -05:00
										 |  |  |  |   if (ioctl (socket->priv->fd, FIONREAD, &avail) < 0) | 
					
						
							| 
									
										
										
										
											2013-08-24 13:55:06 -04:00
										 |  |  |  |     avail = -1; | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-02-01 14:21:10 +01:00
										 |  |  |  |   if (socket->priv->type == G_SOCKET_TYPE_DATAGRAM) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (G_UNLIKELY (g_once_init_enter (&buf))) | 
					
						
							|  |  |  |  |         g_once_init_leave (&buf, g_malloc (bufsize)); | 
					
						
							| 
									
										
										
										
											2013-08-24 13:55:06 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-01 14:21:10 +01:00
										 |  |  |  |       avail = recv (socket->priv->fd, buf, bufsize, MSG_PEEK); | 
					
						
							|  |  |  |  |       if (avail == -1 && get_socket_errno () == WSAEWOULDBLOCK) | 
					
						
							|  |  |  |  |         avail = 0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (ioctlsocket (socket->priv->fd, FIONREAD, &avail) < 0) | 
					
						
							|  |  |  |  |         avail = -1; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-01-16 14:00:49 +01:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return avail; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  | /* Block on a timed wait for @condition until (@start_time + @timeout).
 | 
					
						
							|  |  |  |  |  * Return %G_IO_ERROR_TIMED_OUT if the timeout is reached; otherwise %TRUE. | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							|  |  |  |  | block_on_timeout (GSocket       *socket, | 
					
						
							|  |  |  |  |                   GIOCondition   condition, | 
					
						
							|  |  |  |  |                   gint64         timeout, | 
					
						
							|  |  |  |  |                   gint64         start_time, | 
					
						
							|  |  |  |  |                   GCancellable  *cancellable, | 
					
						
							|  |  |  |  |                   GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint64 wait_timeout = -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (timeout != 0, TRUE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* check if we've timed out or how much time to wait at most */ | 
					
						
							|  |  |  |  |   if (timeout >= 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       gint64 elapsed = g_get_monotonic_time () - start_time; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (elapsed >= timeout) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           g_set_error_literal (error, | 
					
						
							|  |  |  |  |                                G_IO_ERROR, G_IO_ERROR_TIMED_OUT, | 
					
						
							|  |  |  |  |                                _("Socket I/O timed out")); | 
					
						
							|  |  |  |  |           return FALSE; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       wait_timeout = timeout - elapsed; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_condition_timed_wait (socket, condition, wait_timeout, | 
					
						
							|  |  |  |  |                                         cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gssize | 
					
						
							|  |  |  |  | g_socket_receive_with_timeout (GSocket       *socket, | 
					
						
							|  |  |  |  |                                guint8        *buffer, | 
					
						
							|  |  |  |  |                                gsize          size, | 
					
						
							|  |  |  |  |                                gint64         timeout, | 
					
						
							|  |  |  |  |                                GCancellable  *cancellable, | 
					
						
							|  |  |  |  |                                GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gssize ret; | 
					
						
							|  |  |  |  |   gint64 start_time; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket) && buffer != NULL, -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   start_time = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   while (1) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |       win32_unset_event_mask (socket, FD_READ); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |       if ((ret = recv (socket->priv->fd, buffer, size, 0)) < 0) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  if (errsv == EINTR) | 
					
						
							|  |  |  |  | 	    continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef WSAEWOULDBLOCK
 | 
					
						
							|  |  |  |  |           if (errsv == WSAEWOULDBLOCK) | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |           if (errsv == EWOULDBLOCK || | 
					
						
							|  |  |  |  |               errsv == EAGAIN) | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               if (timeout != 0) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   if (!block_on_timeout (socket, G_IO_IN, timeout, start_time, | 
					
						
							|  |  |  |  |                                          cancellable, error)) | 
					
						
							|  |  |  |  |                     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   continue; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  socket_set_error_lazy (error, errsv, _("Error receiving data: %s")); | 
					
						
							|  |  |  |  | 	  return -1; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_receive: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2013-01-13 20:49:15 +01:00
										 |  |  |  |  * @buffer: (array length=size) (element-type guint8): a buffer to | 
					
						
							|  |  |  |  |  *     read data into (which should be at least @size bytes long). | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * @size: the number of bytes you want to read from the socket | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Receive data (up to @size bytes) from a socket. This is mainly used by | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * connection-oriented sockets; it is identical to g_socket_receive_from() | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * with @address set to %NULL. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * For %G_SOCKET_TYPE_DATAGRAM and %G_SOCKET_TYPE_SEQPACKET sockets, | 
					
						
							|  |  |  |  |  * g_socket_receive() will always read either 0 or 1 complete messages from | 
					
						
							|  |  |  |  |  * the socket. If the received message is too large to fit in @buffer, then | 
					
						
							|  |  |  |  |  * the data beyond @size bytes will be discarded, without any explicit | 
					
						
							|  |  |  |  |  * indication that this has occurred. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * For %G_SOCKET_TYPE_STREAM sockets, g_socket_receive() can return any | 
					
						
							|  |  |  |  |  * number of bytes, up to @size. If more than @size bytes have been | 
					
						
							|  |  |  |  |  * received, the additional data will be returned in future calls to | 
					
						
							|  |  |  |  |  * g_socket_receive(). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-04-09 12:47:32 -04:00
										 |  |  |  |  * If the socket is in blocking mode the call will block until there | 
					
						
							|  |  |  |  |  * is some data to receive, the connection is closed, or there is an | 
					
						
							|  |  |  |  |  * error. If there is no data available and the socket is in | 
					
						
							|  |  |  |  |  * non-blocking mode, a %G_IO_ERROR_WOULD_BLOCK error will be | 
					
						
							|  |  |  |  |  * returned. To be notified when data is available, wait for the | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * %G_IO_IN condition. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * On error -1 is returned and @error is set accordingly. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-04-09 12:47:32 -04:00
										 |  |  |  |  * Returns: Number of bytes read, or 0 if the connection was closed by | 
					
						
							|  |  |  |  |  * the peer, or -1 on error | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gssize | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | g_socket_receive (GSocket       *socket, | 
					
						
							|  |  |  |  | 		  gchar         *buffer, | 
					
						
							|  |  |  |  | 		  gsize          size, | 
					
						
							|  |  |  |  | 		  GCancellable  *cancellable, | 
					
						
							|  |  |  |  | 		  GError       **error) | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   return g_socket_receive_with_timeout (socket, (guint8 *) buffer, size, | 
					
						
							|  |  |  |  |                                         socket->priv->blocking ? -1 : 0, | 
					
						
							|  |  |  |  |                                         cancellable, error); | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_receive_with_blocking: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2013-01-13 20:49:15 +01:00
										 |  |  |  |  * @buffer: (array length=size) (element-type guint8): a buffer to | 
					
						
							|  |  |  |  |  *     read data into (which should be at least @size bytes long). | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  |  * @size: the number of bytes you want to read from the socket | 
					
						
							|  |  |  |  |  * @blocking: whether to do blocking or non-blocking I/O | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This behaves exactly the same as g_socket_receive(), except that | 
					
						
							|  |  |  |  |  * the choice of blocking or non-blocking behavior is determined by | 
					
						
							|  |  |  |  |  * the @blocking argument rather than by @socket's properties. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-04-09 12:47:32 -04:00
										 |  |  |  |  * Returns: Number of bytes read, or 0 if the connection was closed by | 
					
						
							|  |  |  |  |  * the peer, or -1 on error | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.26 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gssize | 
					
						
							|  |  |  |  | g_socket_receive_with_blocking (GSocket       *socket, | 
					
						
							|  |  |  |  | 				gchar         *buffer, | 
					
						
							|  |  |  |  | 				gsize          size, | 
					
						
							|  |  |  |  | 				gboolean       blocking, | 
					
						
							|  |  |  |  | 				GCancellable  *cancellable, | 
					
						
							|  |  |  |  | 				GError       **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   return g_socket_receive_with_timeout (socket, (guint8 *) buffer, size, | 
					
						
							|  |  |  |  |                                         blocking ? -1 : 0, cancellable, error); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_receive_from: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @address: (out) (optional): a pointer to a #GSocketAddress | 
					
						
							| 
									
										
										
										
											2012-01-07 09:54:53 -08:00
										 |  |  |  |  *     pointer, or %NULL | 
					
						
							|  |  |  |  |  * @buffer: (array length=size) (element-type guint8): a buffer to | 
					
						
							|  |  |  |  |  *     read data into (which should be at least @size bytes long). | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * @size: the number of bytes you want to read from the socket | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Receive data (up to @size bytes) from a socket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If @address is non-%NULL then @address will be set equal to the | 
					
						
							|  |  |  |  |  * source address of the received packet. | 
					
						
							|  |  |  |  |  * @address is owned by the caller. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * See g_socket_receive() for additional information. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-04-09 12:47:32 -04:00
										 |  |  |  |  * Returns: Number of bytes read, or 0 if the connection was closed by | 
					
						
							|  |  |  |  |  * the peer, or -1 on error | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gssize | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_receive_from (GSocket         *socket, | 
					
						
							|  |  |  |  | 		       GSocketAddress **address, | 
					
						
							|  |  |  |  | 		       gchar           *buffer, | 
					
						
							|  |  |  |  | 		       gsize            size, | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 		       GCancellable    *cancellable, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 		       GError         **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   GInputVector v; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   v.buffer = buffer; | 
					
						
							|  |  |  |  |   v.size = size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_receive_message (socket, | 
					
						
							|  |  |  |  | 				   address, | 
					
						
							|  |  |  |  | 				   &v, 1, | 
					
						
							|  |  |  |  | 				   NULL, 0, NULL, | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 				   cancellable, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 				   error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 09:59:05 -04:00
										 |  |  |  | /* See the comment about SIGPIPE above. */ | 
					
						
							| 
									
										
										
										
											2009-08-19 12:12:06 -04:00
										 |  |  |  | #ifdef MSG_NOSIGNAL
 | 
					
						
							|  |  |  |  | #define G_SOCKET_DEFAULT_SEND_FLAGS MSG_NOSIGNAL
 | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  | #define G_SOCKET_DEFAULT_SEND_FLAGS 0
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  | static gssize | 
					
						
							|  |  |  |  | g_socket_send_with_timeout (GSocket       *socket, | 
					
						
							|  |  |  |  |                             const guint8  *buffer, | 
					
						
							|  |  |  |  |                             gsize          size, | 
					
						
							|  |  |  |  |                             gint64         timeout, | 
					
						
							|  |  |  |  |                             GCancellable  *cancellable, | 
					
						
							|  |  |  |  |                             GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gssize ret; | 
					
						
							|  |  |  |  |   gint64 start_time; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket) && buffer != NULL, -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   start_time = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   while (1) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |       win32_unset_event_mask (socket, FD_WRITE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |       if ((ret = send (socket->priv->fd, buffer, size, G_SOCKET_DEFAULT_SEND_FLAGS)) < 0) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  if (errsv == EINTR) | 
					
						
							|  |  |  |  | 	    continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef WSAEWOULDBLOCK
 | 
					
						
							|  |  |  |  |           if (errsv == WSAEWOULDBLOCK) | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |           if (errsv == EWOULDBLOCK || | 
					
						
							|  |  |  |  |               errsv == EAGAIN) | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |               if (timeout != 0) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                   if (!block_on_timeout (socket, G_IO_OUT, timeout, start_time, | 
					
						
							|  |  |  |  |                                          cancellable, error)) | 
					
						
							|  |  |  |  |                     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                   continue; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  socket_set_error_lazy (error, errsv, _("Error sending data: %s")); | 
					
						
							|  |  |  |  | 	  return -1; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_send: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2012-01-07 09:54:53 -08:00
										 |  |  |  |  * @buffer: (array length=size) (element-type guint8): the buffer | 
					
						
							|  |  |  |  |  *     containing the data to send. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @size: the number of bytes to send | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * Tries to send @size bytes from @buffer on the socket. This is | 
					
						
							|  |  |  |  |  * mainly used by connection-oriented sockets; it is identical to | 
					
						
							|  |  |  |  |  * g_socket_send_to() with @address set to %NULL. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * If the socket is in blocking mode the call will block until there is | 
					
						
							|  |  |  |  |  * space for the data in the socket queue. If there is no space available | 
					
						
							|  |  |  |  |  * and the socket is in non-blocking mode a %G_IO_ERROR_WOULD_BLOCK error | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * will be returned. To be notified when space is available, wait for the | 
					
						
							|  |  |  |  |  * %G_IO_OUT condition. Note though that you may still receive | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_WOULD_BLOCK from g_socket_send() even if you were previously | 
					
						
							|  |  |  |  |  * notified of a %G_IO_OUT condition. (On Windows in particular, this is | 
					
						
							|  |  |  |  |  * very common due to the way the underlying APIs work.) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * On error -1 is returned and @error is set accordingly. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * Returns: Number of bytes written (which may be less than @size), or -1 | 
					
						
							|  |  |  |  |  * on error | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gssize | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | g_socket_send (GSocket       *socket, | 
					
						
							|  |  |  |  | 	       const gchar   *buffer, | 
					
						
							|  |  |  |  | 	       gsize          size, | 
					
						
							|  |  |  |  | 	       GCancellable  *cancellable, | 
					
						
							|  |  |  |  | 	       GError       **error) | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   return g_socket_send_with_blocking (socket, buffer, size, | 
					
						
							|  |  |  |  | 				      socket->priv->blocking, | 
					
						
							|  |  |  |  | 				      cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_send_with_blocking: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2012-01-07 09:54:53 -08:00
										 |  |  |  |  * @buffer: (array length=size) (element-type guint8): the buffer | 
					
						
							|  |  |  |  |  *     containing the data to send. | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  |  * @size: the number of bytes to send | 
					
						
							|  |  |  |  |  * @blocking: whether to do blocking or non-blocking I/O | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2010-06-24 13:09:14 -04:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This behaves exactly the same as g_socket_send(), except that | 
					
						
							|  |  |  |  |  * the choice of blocking or non-blocking behavior is determined by | 
					
						
							|  |  |  |  |  * the @blocking argument rather than by @socket's properties. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: Number of bytes written (which may be less than @size), or -1 | 
					
						
							|  |  |  |  |  * on error | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.26 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gssize | 
					
						
							|  |  |  |  | g_socket_send_with_blocking (GSocket       *socket, | 
					
						
							|  |  |  |  | 			     const gchar   *buffer, | 
					
						
							|  |  |  |  | 			     gsize          size, | 
					
						
							|  |  |  |  | 			     gboolean       blocking, | 
					
						
							|  |  |  |  | 			     GCancellable  *cancellable, | 
					
						
							|  |  |  |  | 			     GError       **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   return g_socket_send_with_timeout (socket, (const guint8 *) buffer, size, | 
					
						
							|  |  |  |  |                                      blocking ? -1 : 0, cancellable, error); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_send_to: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @address: (nullable): a #GSocketAddress, or %NULL | 
					
						
							| 
									
										
										
										
											2012-01-07 09:54:53 -08:00
										 |  |  |  |  * @buffer: (array length=size) (element-type guint8): the buffer | 
					
						
							|  |  |  |  |  *     containing the data to send. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @size: the number of bytes to send | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Tries to send @size bytes from @buffer to @address. If @address is | 
					
						
							|  |  |  |  |  * %NULL then the message is sent to the default receiver (set by | 
					
						
							|  |  |  |  |  * g_socket_connect()). | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * See g_socket_send() for additional information. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * Returns: Number of bytes written (which may be less than @size), or -1 | 
					
						
							|  |  |  |  |  * on error | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gssize | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_send_to (GSocket         *socket, | 
					
						
							|  |  |  |  | 		  GSocketAddress  *address, | 
					
						
							|  |  |  |  | 		  const gchar     *buffer, | 
					
						
							|  |  |  |  | 		  gsize            size, | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 		  GCancellable    *cancellable, | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | 		  GError         **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   GOutputVector v; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   v.buffer = buffer; | 
					
						
							|  |  |  |  |   v.size = size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_send_message (socket, | 
					
						
							|  |  |  |  | 				address, | 
					
						
							|  |  |  |  | 				&v, 1, | 
					
						
							|  |  |  |  | 				NULL, 0, | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 				0, | 
					
						
							|  |  |  |  | 				cancellable, | 
					
						
							|  |  |  |  | 				error); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_shutdown: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @shutdown_read: whether to shut down the read side | 
					
						
							|  |  |  |  |  * @shutdown_write: whether to shut down the write side | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-10-04 11:34:36 +01:00
										 |  |  |  |  * Shut down part or all of a full-duplex connection. | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-08-29 14:49:32 -04:00
										 |  |  |  |  * If @shutdown_read is %TRUE then the receiving side of the connection | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  |  * is shut down, and further reading is disallowed. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If @shutdown_write is %TRUE then the sending side of the connection | 
					
						
							|  |  |  |  |  * is shut down, and further writing is disallowed. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * It is allowed for both @shutdown_read and @shutdown_write to be %TRUE. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-10-04 11:34:36 +01:00
										 |  |  |  |  * One example where it is useful to shut down only one side of a connection is | 
					
						
							|  |  |  |  |  * graceful disconnect for TCP connections where you close the sending side, | 
					
						
							|  |  |  |  |  * then wait for the other side to close the connection, thus ensuring that the | 
					
						
							|  |  |  |  |  * other side saw all sent data. | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE on success, %FALSE on error | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  | gboolean | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_shutdown (GSocket   *socket, | 
					
						
							|  |  |  |  | 		   gboolean   shutdown_read, | 
					
						
							|  |  |  |  | 		   gboolean   shutdown_write, | 
					
						
							|  |  |  |  | 		   GError   **error) | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   int how; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), TRUE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-28 15:13:11 +01:00
										 |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* Do nothing? */ | 
					
						
							|  |  |  |  |   if (!shutdown_read && !shutdown_write) | 
					
						
							|  |  |  |  |     return TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   if (shutdown_read && shutdown_write) | 
					
						
							|  |  |  |  |     how = SHUT_RDWR; | 
					
						
							|  |  |  |  |   else if (shutdown_read) | 
					
						
							|  |  |  |  |     how = SHUT_RD; | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     how = SHUT_WR; | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   if (shutdown_read && shutdown_write) | 
					
						
							|  |  |  |  |     how = SD_BOTH; | 
					
						
							|  |  |  |  |   else if (shutdown_read) | 
					
						
							|  |  |  |  |     how = SD_RECEIVE; | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     how = SD_SEND; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (shutdown (socket->priv->fd, how) != 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  |       g_set_error (error, G_IO_ERROR, socket_io_error_from_errno (errsv), | 
					
						
							| 
									
										
										
										
											2011-05-28 15:06:45 +01:00
										 |  |  |  | 		   _("Unable to shutdown socket: %s"), socket_strerror (errsv)); | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |   if (shutdown_read) | 
					
						
							|  |  |  |  |     socket->priv->connected_read = FALSE; | 
					
						
							|  |  |  |  |   if (shutdown_write) | 
					
						
							|  |  |  |  |     socket->priv->connected_write = FALSE; | 
					
						
							| 
									
										
										
										
											2009-05-19 11:52:33 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_close: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Closes the socket, shutting down any active connection. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  * Closing a socket does not wait for all outstanding I/O operations | 
					
						
							|  |  |  |  |  * to finish, so the caller should not rely on them to be guaranteed | 
					
						
							|  |  |  |  |  * to complete even if the close returns with no error. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  * Once the socket is closed, all other operations will return | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * %G_IO_ERROR_CLOSED. Closing a socket multiple times will not | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  * return an error. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sockets will be automatically closed when the last reference | 
					
						
							|  |  |  |  |  * is dropped, but you might want to call this function to make sure | 
					
						
							|  |  |  |  |  * resources are released as early as possible. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * Beware that due to the way that TCP works, it is possible for | 
					
						
							|  |  |  |  |  * recently-sent data to be lost if either you close a socket while the | 
					
						
							|  |  |  |  |  * %G_IO_IN condition is set, or else if the remote connection tries to | 
					
						
							|  |  |  |  |  * send something to you after you close the socket but before it has | 
					
						
							|  |  |  |  |  * finished reading all of the data you sent. There is no easy generic | 
					
						
							|  |  |  |  |  * way to avoid this problem; the easiest fix is to design the network | 
					
						
							|  |  |  |  |  * protocol such that the client will never send data "out of turn". | 
					
						
							|  |  |  |  |  * Another solution is for the server to half-close the connection by | 
					
						
							|  |  |  |  |  * calling g_socket_shutdown() with only the @shutdown_write flag set, | 
					
						
							|  |  |  |  |  * and then wait for the client to notice this and close its side of the | 
					
						
							|  |  |  |  |  * connection, after which the server can safely call g_socket_close(). | 
					
						
							|  |  |  |  |  * (This is what #GTcpConnection does if you call | 
					
						
							|  |  |  |  |  * g_tcp_connection_set_graceful_disconnect(). But of course, this | 
					
						
							|  |  |  |  |  * only works if the client will close its connection after the server | 
					
						
							|  |  |  |  |  * does.) | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Returns: %TRUE on success, %FALSE on error | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_close (GSocket  *socket, | 
					
						
							|  |  |  |  | 		GError  **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   int res; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), TRUE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->closed) | 
					
						
							|  |  |  |  |     return TRUE; /* Multiple close not an error */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-28 15:13:11 +01:00
										 |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   while (1) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |       res = closesocket (socket->priv->fd); | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |       res = close (socket->priv->fd); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |       if (res == -1) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  if (errsv == EINTR) | 
					
						
							|  |  |  |  | 	    continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	  g_set_error (error, G_IO_ERROR, | 
					
						
							|  |  |  |  | 		       socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 		       _("Error closing socket: %s"), | 
					
						
							|  |  |  |  | 		       socket_strerror (errsv)); | 
					
						
							|  |  |  |  | 	  return FALSE; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  |       break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-04 10:40:50 +03:00
										 |  |  |  |   socket->priv->fd = -1; | 
					
						
							| 
									
										
										
										
											2015-08-17 20:03:47 +01:00
										 |  |  |  |   socket->priv->connected_read = FALSE; | 
					
						
							|  |  |  |  |   socket->priv->connected_write = FALSE; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   socket->priv->closed = TRUE; | 
					
						
							| 
									
										
										
										
											2010-06-25 14:28:02 -04:00
										 |  |  |  |   if (socket->priv->remote_address) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_object_unref (socket->priv->remote_address); | 
					
						
							|  |  |  |  |       socket->priv->remote_address = NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_is_closed: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Checks whether a socket is closed. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE if socket is closed, %FALSE otherwise | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_is_closed (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   return socket->priv->closed; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  | /* Broken source, used on errors */ | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | broken_dispatch (GSource     *source, | 
					
						
							|  |  |  |  | 		 GSourceFunc  callback, | 
					
						
							|  |  |  |  | 		 gpointer     user_data) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GSourceFuncs broken_funcs = | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-03 08:10:31 -03:00
										 |  |  |  |   NULL, | 
					
						
							|  |  |  |  |   NULL, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   broken_dispatch, | 
					
						
							|  |  |  |  |   NULL | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gint | 
					
						
							|  |  |  |  | network_events_for_condition (GIOCondition condition) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   int event_mask = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (condition & G_IO_IN) | 
					
						
							|  |  |  |  |     event_mask |= (FD_READ | FD_ACCEPT); | 
					
						
							|  |  |  |  |   if (condition & G_IO_OUT) | 
					
						
							|  |  |  |  |     event_mask |= (FD_WRITE | FD_CONNECT); | 
					
						
							|  |  |  |  |   event_mask |= FD_CLOSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return event_mask; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | ensure_event (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (socket->priv->event == WSA_INVALID_EVENT) | 
					
						
							|  |  |  |  |     socket->priv->event = WSACreateEvent(); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | update_select_events (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   int event_mask; | 
					
						
							|  |  |  |  |   GIOCondition *ptr; | 
					
						
							|  |  |  |  |   GList *l; | 
					
						
							|  |  |  |  |   WSAEVENT event; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   ensure_event (socket); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   event_mask = 0; | 
					
						
							|  |  |  |  |   for (l = socket->priv->requested_conditions; l != NULL; l = l->next) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       ptr = l->data; | 
					
						
							|  |  |  |  |       event_mask |= network_events_for_condition (*ptr); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (event_mask != socket->priv->selected_events) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       /* If no events selected, disable event so we can unset
 | 
					
						
							|  |  |  |  | 	 nonblocking mode */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (event_mask == 0) | 
					
						
							|  |  |  |  | 	event = NULL; | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  | 	event = socket->priv->event; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (WSAEventSelect (socket->priv->fd, event, event_mask) == 0) | 
					
						
							|  |  |  |  | 	socket->priv->selected_events = event_mask; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | add_condition_watch (GSocket      *socket, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		     GIOCondition *condition) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-28 16:43:44 +02:00
										 |  |  |  |   g_mutex_lock (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   g_assert (g_list_find (socket->priv->requested_conditions, condition) == NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket->priv->requested_conditions = | 
					
						
							|  |  |  |  |     g_list_prepend (socket->priv->requested_conditions, condition); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   update_select_events (socket); | 
					
						
							| 
									
										
										
										
											2013-07-28 16:43:44 +02:00
										 |  |  |  |   g_mutex_unlock (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | remove_condition_watch (GSocket      *socket, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 			GIOCondition *condition) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-07-28 16:43:44 +02:00
										 |  |  |  |   g_mutex_lock (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   g_assert (g_list_find (socket->priv->requested_conditions, condition) != NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket->priv->requested_conditions = | 
					
						
							|  |  |  |  |     g_list_remove (socket->priv->requested_conditions, condition); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   update_select_events (socket); | 
					
						
							| 
									
										
										
										
											2013-07-28 16:43:44 +02:00
										 |  |  |  |   g_mutex_unlock (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GIOCondition | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  | update_condition_unlocked (GSocket *socket) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   WSANETWORKEVENTS events; | 
					
						
							|  |  |  |  |   GIOCondition condition; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (WSAEnumNetworkEvents (socket->priv->fd, | 
					
						
							|  |  |  |  | 			    socket->priv->event, | 
					
						
							|  |  |  |  | 			    &events) == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       socket->priv->current_events |= events.lNetworkEvents; | 
					
						
							|  |  |  |  |       if (events.lNetworkEvents & FD_WRITE && | 
					
						
							|  |  |  |  | 	  events.iErrorCode[FD_WRITE_BIT] != 0) | 
					
						
							|  |  |  |  | 	socket->priv->current_errors |= FD_WRITE; | 
					
						
							|  |  |  |  |       if (events.lNetworkEvents & FD_CONNECT && | 
					
						
							|  |  |  |  | 	  events.iErrorCode[FD_CONNECT_BIT] != 0) | 
					
						
							|  |  |  |  | 	socket->priv->current_errors |= FD_CONNECT; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   condition = 0; | 
					
						
							|  |  |  |  |   if (socket->priv->current_events & (FD_READ | FD_ACCEPT)) | 
					
						
							|  |  |  |  |     condition |= G_IO_IN; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-10 02:02:29 +01:00
										 |  |  |  |   if (socket->priv->current_events & FD_CLOSE) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int r, errsv, buffer; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       r = recv (socket->priv->fd, &buffer, sizeof (buffer), MSG_PEEK); | 
					
						
							|  |  |  |  |       if (r < 0) | 
					
						
							|  |  |  |  |           errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (r > 0 || | 
					
						
							|  |  |  |  |           (r < 0 && errsv == WSAENOTCONN)) | 
					
						
							|  |  |  |  |         condition |= G_IO_IN; | 
					
						
							|  |  |  |  |       else if (r == 0 || | 
					
						
							|  |  |  |  |                (r < 0 && (errsv == WSAESHUTDOWN || errsv == WSAECONNRESET || | 
					
						
							|  |  |  |  |                           errsv == WSAECONNABORTED || errsv == WSAENETRESET))) | 
					
						
							|  |  |  |  |         condition |= G_IO_HUP; | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  |         condition |= G_IO_ERR; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->closed) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     condition |= G_IO_HUP; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* Never report both G_IO_OUT and HUP, these are
 | 
					
						
							|  |  |  |  |      mutually exclusive (can't write to a closed socket) */ | 
					
						
							|  |  |  |  |   if ((condition & G_IO_HUP) == 0 && | 
					
						
							|  |  |  |  |       socket->priv->current_events & FD_WRITE) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (socket->priv->current_errors & FD_WRITE) | 
					
						
							|  |  |  |  | 	condition |= G_IO_ERR; | 
					
						
							|  |  |  |  |       else | 
					
						
							|  |  |  |  | 	condition |= G_IO_OUT; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       if (socket->priv->current_events & FD_CONNECT) | 
					
						
							|  |  |  |  | 	{ | 
					
						
							|  |  |  |  | 	  if (socket->priv->current_errors & FD_CONNECT) | 
					
						
							|  |  |  |  | 	    condition |= (G_IO_HUP | G_IO_ERR); | 
					
						
							|  |  |  |  | 	  else | 
					
						
							|  |  |  |  | 	    condition |= G_IO_OUT; | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return condition; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static GIOCondition | 
					
						
							|  |  |  |  | update_condition (GSocket *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GIOCondition res; | 
					
						
							|  |  |  |  |   g_mutex_lock (&socket->priv->win32_source_lock); | 
					
						
							|  |  |  |  |   res = update_condition_unlocked (socket); | 
					
						
							|  |  |  |  |   g_mutex_unlock (&socket->priv->win32_source_lock); | 
					
						
							|  |  |  |  |   return res; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | typedef struct { | 
					
						
							|  |  |  |  |   GSource       source; | 
					
						
							| 
									
										
										
										
											2014-02-13 16:37:49 -05:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   GPollFD       pollfd; | 
					
						
							| 
									
										
										
										
											2014-02-13 16:37:49 -05:00
										 |  |  |  | #else
 | 
					
						
							|  |  |  |  |   gpointer      fd_tag; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   GSocket      *socket; | 
					
						
							|  |  |  |  |   GIOCondition  condition; | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | } GSocketSource; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 17:07:07 -05:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2014-02-13 17:07:07 -05:00
										 |  |  |  | socket_source_prepare_win32 (GSource *source, | 
					
						
							|  |  |  |  |                              gint    *timeout) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   GSocketSource *socket_source = (GSocketSource *)source; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 16:59:28 -05:00
										 |  |  |  |   *timeout = -1; | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 17:07:07 -05:00
										 |  |  |  |   return (update_condition (socket_source->socket) & socket_source->condition) != 0; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2014-02-13 17:07:07 -05:00
										 |  |  |  | socket_source_check_win32 (GSource *source) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   int timeout; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 17:07:07 -05:00
										 |  |  |  |   return socket_source_prepare_win32 (source, &timeout); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-13 17:07:07 -05:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static gboolean | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | socket_source_dispatch (GSource     *source, | 
					
						
							|  |  |  |  | 			GSourceFunc  callback, | 
					
						
							|  |  |  |  | 			gpointer     user_data) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   GSocketSourceFunc func = (GSocketSourceFunc)callback; | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   GSocketSource *socket_source = (GSocketSource *)source; | 
					
						
							| 
									
										
										
										
											2012-01-16 15:49:50 +01:00
										 |  |  |  |   GSocket *socket = socket_source->socket; | 
					
						
							| 
									
										
										
										
											2014-02-13 16:59:28 -05:00
										 |  |  |  |   gint64 timeout; | 
					
						
							| 
									
										
										
										
											2014-02-13 16:58:34 -05:00
										 |  |  |  |   guint events; | 
					
						
							| 
									
										
										
										
											2012-01-16 15:49:50 +01:00
										 |  |  |  |   gboolean ret; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-03 15:20:12 +02:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							| 
									
										
										
										
											2014-02-13 16:58:34 -05:00
										 |  |  |  |   events = update_condition (socket_source->socket); | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-02-13 16:37:49 -05:00
										 |  |  |  |   events = g_source_query_unix_fd (source, socket_source->fd_tag); | 
					
						
							| 
									
										
										
										
											2011-01-03 15:20:12 +02:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-02-13 16:58:34 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 16:59:28 -05:00
										 |  |  |  |   timeout = g_source_get_ready_time (source); | 
					
						
							|  |  |  |  |   if (timeout >= 0 && timeout < g_source_get_time (source)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       socket->priv->timed_out = TRUE; | 
					
						
							|  |  |  |  |       events |= (G_IO_IN | G_IO_OUT); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-03 15:20:12 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 16:58:34 -05:00
										 |  |  |  |   ret = (*func) (socket, events & socket_source->condition, user_data); | 
					
						
							| 
									
										
										
										
											2012-01-16 15:49:50 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->timeout) | 
					
						
							| 
									
										
										
										
											2014-02-13 16:59:28 -05:00
										 |  |  |  |     g_source_set_ready_time (source, g_get_monotonic_time () + socket->priv->timeout * 1000000); | 
					
						
							| 
									
										
										
										
											2012-01-16 15:49:50 +01:00
										 |  |  |  |   else | 
					
						
							| 
									
										
										
										
											2014-02-13 16:59:28 -05:00
										 |  |  |  |     g_source_set_ready_time (source, -1); | 
					
						
							| 
									
										
										
										
											2012-01-16 15:49:50 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | socket_source_finalize (GSource *source) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   GSocketSource *socket_source = (GSocketSource *)source; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   GSocket *socket; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   socket = socket_source->socket; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   remove_condition_watch (socket, &socket_source->condition); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_object_unref (socket); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-07 11:05:26 -05:00
										 |  |  |  | static gboolean | 
					
						
							|  |  |  |  | socket_source_closure_callback (GSocket      *socket, | 
					
						
							|  |  |  |  | 				GIOCondition  condition, | 
					
						
							|  |  |  |  | 				gpointer      data) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GClosure *closure = data; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-30 17:08:15 +01:00
										 |  |  |  |   GValue params[2] = { G_VALUE_INIT, G_VALUE_INIT }; | 
					
						
							|  |  |  |  |   GValue result_value = G_VALUE_INIT; | 
					
						
							| 
									
										
										
										
											2010-11-07 11:05:26 -05:00
										 |  |  |  |   gboolean result; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_value_init (&result_value, G_TYPE_BOOLEAN); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_value_init (¶ms[0], G_TYPE_SOCKET); | 
					
						
							|  |  |  |  |   g_value_set_object (¶ms[0], socket); | 
					
						
							|  |  |  |  |   g_value_init (¶ms[1], G_TYPE_IO_CONDITION); | 
					
						
							|  |  |  |  |   g_value_set_flags (¶ms[1], condition); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_closure_invoke (closure, &result_value, 2, params, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   result = g_value_get_boolean (&result_value); | 
					
						
							|  |  |  |  |   g_value_unset (&result_value); | 
					
						
							|  |  |  |  |   g_value_unset (¶ms[0]); | 
					
						
							|  |  |  |  |   g_value_unset (¶ms[1]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return result; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | static GSourceFuncs socket_source_funcs = | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-13 17:07:07 -05:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   socket_source_prepare_win32, | 
					
						
							|  |  |  |  |   socket_source_check_win32, | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   NULL, NULL, /* check, prepare */ | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   socket_source_dispatch, | 
					
						
							| 
									
										
										
										
											2010-11-07 11:05:26 -05:00
										 |  |  |  |   socket_source_finalize, | 
					
						
							|  |  |  |  |   (GSourceFunc)socket_source_closure_callback, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static GSource * | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | socket_source_new (GSocket      *socket, | 
					
						
							|  |  |  |  | 		   GIOCondition  condition, | 
					
						
							|  |  |  |  | 		   GCancellable *cancellable) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   GSource *source; | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   GSocketSource *socket_source; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   ensure_event (socket); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->event == WSA_INVALID_EVENT) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_warning ("Failed to create WSAEvent"); | 
					
						
							|  |  |  |  |       return g_source_new (&broken_funcs, sizeof (GSource)); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-29 09:25:42 -04:00
										 |  |  |  |   condition |= G_IO_HUP | G_IO_ERR | G_IO_NVAL; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   source = g_source_new (&socket_source_funcs, sizeof (GSocketSource)); | 
					
						
							| 
									
										
										
										
											2010-07-10 20:09:34 -04:00
										 |  |  |  |   g_source_set_name (source, "GSocket"); | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   socket_source = (GSocketSource *)source; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   socket_source->socket = g_object_ref (socket); | 
					
						
							|  |  |  |  |   socket_source->condition = condition; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 16:34:29 -05:00
										 |  |  |  |   if (cancellable) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-02-13 16:34:29 -05:00
										 |  |  |  |       GSource *cancellable_source; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       cancellable_source = g_cancellable_source_new (cancellable); | 
					
						
							|  |  |  |  |       g_source_add_child_source (source, cancellable_source); | 
					
						
							|  |  |  |  |       g_source_set_dummy_callback (cancellable_source); | 
					
						
							|  |  |  |  |       g_source_unref (cancellable_source); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   add_condition_watch (socket, &socket_source->condition); | 
					
						
							|  |  |  |  |   socket_source->pollfd.fd = (gintptr) socket->priv->event; | 
					
						
							|  |  |  |  |   socket_source->pollfd.events = condition; | 
					
						
							|  |  |  |  |   socket_source->pollfd.revents = 0; | 
					
						
							|  |  |  |  |   g_source_add_poll (source, &socket_source->pollfd); | 
					
						
							| 
									
										
										
										
											2014-02-13 16:37:49 -05:00
										 |  |  |  | #else
 | 
					
						
							|  |  |  |  |   socket_source->fd_tag = g_source_add_unix_fd (source, socket->priv->fd, condition); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |   if (socket->priv->timeout) | 
					
						
							| 
									
										
										
										
											2014-02-13 16:59:28 -05:00
										 |  |  |  |     g_source_set_ready_time (source, g_get_monotonic_time () + socket->priv->timeout * 1000000); | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |   else | 
					
						
							| 
									
										
										
										
											2014-02-13 16:59:28 -05:00
										 |  |  |  |     g_source_set_ready_time (source, -1); | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   return source; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2010-12-29 13:51:44 +01:00
										 |  |  |  |  * g_socket_create_source: (skip) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @condition: a #GIOCondition mask to monitor | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-06-23 10:12:08 +01:00
										 |  |  |  |  * Creates a #GSource that can be attached to a %GMainContext to monitor | 
					
						
							|  |  |  |  |  * for the availability of the specified @condition on the socket. The #GSource | 
					
						
							|  |  |  |  |  * keeps a reference to the @socket. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * The callback on the source is of the #GSocketSourceFunc type. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |  * It is meaningless to specify %G_IO_ERR or %G_IO_HUP in @condition; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * these conditions will always be reported output if they are true. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @cancellable if not %NULL can be used to cancel the source, which will | 
					
						
							| 
									
										
										
										
											2009-05-20 11:30:43 +02:00
										 |  |  |  |  * cause the source to trigger, reporting the current condition (which | 
					
						
							|  |  |  |  |  * is likely 0 unless cancellation happened at the same time as a | 
					
						
							|  |  |  |  |  * condition change). You can check for this in the callback using | 
					
						
							|  |  |  |  |  * g_cancellable_is_cancelled(). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |  * If @socket has a timeout set, and it is reached before @condition | 
					
						
							|  |  |  |  |  * occurs, the source will then trigger anyway, reporting %G_IO_IN or | 
					
						
							|  |  |  |  |  * %G_IO_OUT depending on @condition. However, @socket will have been | 
					
						
							|  |  |  |  |  * marked as having had a timeout, and so the next #GSocket I/O method | 
					
						
							|  |  |  |  |  * you call will then fail with a %G_IO_ERROR_TIMED_OUT. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-09-24 18:24:41 -03:00
										 |  |  |  |  * Returns: (transfer full): a newly allocated %GSource, free with g_source_unref(). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GSource * | 
					
						
							|  |  |  |  | g_socket_create_source (GSocket      *socket, | 
					
						
							|  |  |  |  | 			GIOCondition  condition, | 
					
						
							|  |  |  |  | 			GCancellable *cancellable) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket) && (cancellable == NULL || G_IS_CANCELLABLE (cancellable)), NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-09 09:55:00 -04:00
										 |  |  |  |   return socket_source_new (socket, condition, cancellable); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_condition_check: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @condition: a #GIOCondition mask to check | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  * Checks on the readiness of @socket to perform operations. | 
					
						
							|  |  |  |  |  * The operations specified in @condition are checked for and masked | 
					
						
							|  |  |  |  |  * against the currently-satisfied conditions on @socket. The result | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * is returned. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-04-23 08:21:41 -04:00
										 |  |  |  |  * Note that on Windows, it is possible for an operation to return | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_WOULD_BLOCK even immediately after | 
					
						
							|  |  |  |  |  * g_socket_condition_check() has claimed that the socket is ready for | 
					
						
							|  |  |  |  |  * writing. Rather than calling g_socket_condition_check() and then | 
					
						
							|  |  |  |  |  * writing to the socket if it succeeds, it is generally better to | 
					
						
							|  |  |  |  |  * simply try writing to the socket right away, and try again later if | 
					
						
							|  |  |  |  |  * the initial attempt returns %G_IO_ERROR_WOULD_BLOCK. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * It is meaningless to specify %G_IO_ERR or %G_IO_HUP in condition; | 
					
						
							|  |  |  |  |  * these conditions will always be set in the output if they are true. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This call never blocks. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: the @GIOCondition mask of the current state | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | GIOCondition | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  | g_socket_condition_check (GSocket      *socket, | 
					
						
							|  |  |  |  | 			  GIOCondition  condition) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-01-04 00:13:50 +02:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), 0); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   if (!check_socket (socket, NULL)) | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     GIOCondition current_condition; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     condition |= G_IO_ERR | G_IO_HUP; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     add_condition_watch (socket, &condition); | 
					
						
							|  |  |  |  |     current_condition = update_condition (socket); | 
					
						
							|  |  |  |  |     remove_condition_watch (socket, &condition); | 
					
						
							|  |  |  |  |     return condition & current_condition; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     GPollFD poll_fd; | 
					
						
							|  |  |  |  |     gint result; | 
					
						
							|  |  |  |  |     poll_fd.fd = socket->priv->fd; | 
					
						
							|  |  |  |  |     poll_fd.events = condition; | 
					
						
							| 
									
										
										
										
											2012-11-09 09:22:19 -08:00
										 |  |  |  |     poll_fd.revents = 0; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     do | 
					
						
							|  |  |  |  |       result = g_poll (&poll_fd, 1, 0); | 
					
						
							|  |  |  |  |     while (result == -1 && get_socket_errno () == EINTR); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return poll_fd.revents; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_condition_wait: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @condition: a #GIOCondition mask to wait for | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a #GCancellable, or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: a #GError pointer, or %NULL | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  * Waits for @condition to become true on @socket. When the condition | 
					
						
							|  |  |  |  |  * is met, %TRUE is returned. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |  * If @cancellable is cancelled before the condition is met, or if the | 
					
						
							|  |  |  |  |  * socket has a timeout set and it is reached before the condition is | 
					
						
							|  |  |  |  |  * met, then %FALSE is returned and @error, if non-%NULL, is set to | 
					
						
							|  |  |  |  |  * the appropriate value (%G_IO_ERROR_CANCELLED or | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_TIMED_OUT). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  |  * See also g_socket_condition_timed_wait(). | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * Returns: %TRUE if the condition was met, %FALSE otherwise | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_condition_wait (GSocket       *socket, | 
					
						
							|  |  |  |  | 			 GIOCondition   condition, | 
					
						
							|  |  |  |  | 			 GCancellable  *cancellable, | 
					
						
							|  |  |  |  | 			 GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-01-04 00:13:50 +02:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  |   return g_socket_condition_timed_wait (socket, condition, -1, | 
					
						
							|  |  |  |  | 					cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_condition_timed_wait: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @condition: a #GIOCondition mask to wait for | 
					
						
							|  |  |  |  |  * @timeout: the maximum time (in microseconds) to wait, or -1 | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a #GCancellable, or %NULL | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  |  * @error: a #GError pointer, or %NULL | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Waits for up to @timeout microseconds for @condition to become true | 
					
						
							|  |  |  |  |  * on @socket. If the condition is met, %TRUE is returned. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If @cancellable is cancelled before the condition is met, or if | 
					
						
							|  |  |  |  |  * @timeout (or the socket's #GSocket:timeout) is reached before the | 
					
						
							|  |  |  |  |  * condition is met, then %FALSE is returned and @error, if non-%NULL, | 
					
						
							|  |  |  |  |  * is set to the appropriate value (%G_IO_ERROR_CANCELLED or | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_TIMED_OUT). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If you don't want a timeout, use g_socket_condition_wait(). | 
					
						
							|  |  |  |  |  * (Alternatively, you can pass -1 for @timeout.) | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Note that although @timeout is in microseconds for consistency with | 
					
						
							|  |  |  |  |  * other GLib APIs, this function actually only has millisecond | 
					
						
							|  |  |  |  |  * resolution, and the behavior is undefined if @timeout is not an | 
					
						
							|  |  |  |  |  * exact number of milliseconds. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: %TRUE if the condition was met, %FALSE otherwise | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.32 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_condition_timed_wait (GSocket       *socket, | 
					
						
							|  |  |  |  | 			       GIOCondition   condition, | 
					
						
							|  |  |  |  | 			       gint64         timeout, | 
					
						
							|  |  |  |  | 			       GCancellable  *cancellable, | 
					
						
							|  |  |  |  | 			       GError       **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint64 start_time; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  |     return FALSE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  |   if (socket->priv->timeout && | 
					
						
							|  |  |  |  |       (timeout < 0 || socket->priv->timeout < timeout / G_USEC_PER_SEC)) | 
					
						
							|  |  |  |  |     timeout = socket->priv->timeout * 1000; | 
					
						
							|  |  |  |  |   else if (timeout != -1) | 
					
						
							|  |  |  |  |     timeout = timeout / 1000; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   start_time = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | #ifdef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     GIOCondition current_condition; | 
					
						
							|  |  |  |  |     WSAEVENT events[2]; | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  |     DWORD res; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     GPollFD cancel_fd; | 
					
						
							|  |  |  |  |     int num_events; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Always check these */ | 
					
						
							|  |  |  |  |     condition |=  G_IO_ERR | G_IO_HUP; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     add_condition_watch (socket, &condition); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     num_events = 0; | 
					
						
							|  |  |  |  |     events[num_events++] = socket->priv->event; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-13 20:19:15 +02:00
										 |  |  |  |     if (g_cancellable_make_pollfd (cancellable, &cancel_fd)) | 
					
						
							|  |  |  |  |       events[num_events++] = (WSAEVENT)cancel_fd.fd; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  |     if (timeout == -1) | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |       timeout = WSA_INFINITE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |     g_mutex_lock (&socket->priv->win32_source_lock); | 
					
						
							|  |  |  |  |     current_condition = update_condition_unlocked (socket); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     while ((condition & current_condition) == 0) | 
					
						
							|  |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |         if (!socket->priv->waiting) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             socket->priv->waiting = TRUE; | 
					
						
							|  |  |  |  |             socket->priv->waiting_result = 0; | 
					
						
							|  |  |  |  |             g_mutex_unlock (&socket->priv->win32_source_lock); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             res = WSAWaitForMultipleEvents (num_events, events, FALSE, timeout, FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             g_mutex_lock (&socket->priv->win32_source_lock); | 
					
						
							|  |  |  |  |             socket->priv->waiting = FALSE; | 
					
						
							|  |  |  |  |             socket->priv->waiting_result = res; | 
					
						
							|  |  |  |  |             g_cond_broadcast (&socket->priv->win32_source_cond); | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         else | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             if (timeout != WSA_INFINITE) | 
					
						
							|  |  |  |  |               { | 
					
						
							|  |  |  |  |                 if (!g_cond_wait_until (&socket->priv->win32_source_cond, &socket->priv->win32_source_lock, timeout)) | 
					
						
							|  |  |  |  |                   { | 
					
						
							|  |  |  |  |                     res = WSA_WAIT_TIMEOUT; | 
					
						
							|  |  |  |  |                     break; | 
					
						
							|  |  |  |  |                   } | 
					
						
							|  |  |  |  |                 else | 
					
						
							|  |  |  |  |                   { | 
					
						
							|  |  |  |  |                     res = socket->priv->waiting_result; | 
					
						
							|  |  |  |  |                   } | 
					
						
							|  |  |  |  |               } | 
					
						
							|  |  |  |  |             else | 
					
						
							|  |  |  |  |               { | 
					
						
							|  |  |  |  |                 g_cond_wait (&socket->priv->win32_source_cond, &socket->priv->win32_source_lock); | 
					
						
							|  |  |  |  |                 res = socket->priv->waiting_result; | 
					
						
							|  |  |  |  |               } | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	if (res == WSA_WAIT_FAILED) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	    g_set_error (error, G_IO_ERROR, | 
					
						
							|  |  |  |  | 			 socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 			 _("Waiting for socket condition: %s"), | 
					
						
							|  |  |  |  | 			 socket_strerror (errsv)); | 
					
						
							|  |  |  |  | 	    break; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  | 	else if (res == WSA_WAIT_TIMEOUT) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_TIMED_OUT, | 
					
						
							|  |  |  |  | 				 _("Socket I/O timed out")); | 
					
						
							|  |  |  |  | 	    break; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 	if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  | 	  break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |         current_condition = update_condition_unlocked (socket); | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 	if (timeout != WSA_INFINITE) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    timeout -= (g_get_monotonic_time () - start_time) * 1000; | 
					
						
							|  |  |  |  | 	    if (timeout < 0) | 
					
						
							|  |  |  |  | 	      timeout = 0; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |     g_mutex_unlock (&socket->priv->win32_source_lock); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     remove_condition_watch (socket, &condition); | 
					
						
							| 
									
										
										
										
											2009-08-11 15:12:20 +02:00
										 |  |  |  |     if (num_events > 1) | 
					
						
							|  |  |  |  |       g_cancellable_release_fd (cancellable); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return (condition & current_condition) != 0; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     GPollFD poll_fd[2]; | 
					
						
							|  |  |  |  |     gint result; | 
					
						
							|  |  |  |  |     gint num; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     poll_fd[0].fd = socket->priv->fd; | 
					
						
							|  |  |  |  |     poll_fd[0].events = condition; | 
					
						
							|  |  |  |  |     num = 1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-13 20:19:15 +02:00
										 |  |  |  |     if (g_cancellable_make_pollfd (cancellable, &poll_fd[1])) | 
					
						
							|  |  |  |  |       num++; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  |     while (TRUE) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	result = g_poll (poll_fd, num, timeout); | 
					
						
							|  |  |  |  | 	if (result != -1 || errno != EINTR) | 
					
						
							|  |  |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  | 	if (timeout != -1) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							| 
									
										
										
										
											2014-02-14 15:35:11 -05:00
										 |  |  |  | 	    timeout -= (g_get_monotonic_time () - start_time) / 1000; | 
					
						
							| 
									
										
										
										
											2012-02-13 17:20:04 -05:00
										 |  |  |  | 	    if (timeout < 0) | 
					
						
							|  |  |  |  | 	      timeout = 0; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2009-08-11 15:12:20 +02:00
										 |  |  |  |      | 
					
						
							|  |  |  |  |     if (num > 1) | 
					
						
							|  |  |  |  |       g_cancellable_release_fd (cancellable); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-31 10:29:23 -05:00
										 |  |  |  |     if (result == 0) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_TIMED_OUT, | 
					
						
							|  |  |  |  | 			     _("Socket I/O timed out")); | 
					
						
							|  |  |  |  | 	return FALSE; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return !g_cancellable_set_error_if_cancelled (cancellable, error); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |   #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-05 16:29:05 +08:00
										 |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  | /* Unfortunately these have to be macros rather than inline functions due to
 | 
					
						
							|  |  |  |  |  * using alloca(). */ | 
					
						
							|  |  |  |  | #define output_message_to_msghdr(message, prev_message, msg, prev_msg, error) \
 | 
					
						
							|  |  |  |  | G_STMT_START { \ | 
					
						
							|  |  |  |  |   const GOutputMessage  *_message = (message); \ | 
					
						
							|  |  |  |  |   const GOutputMessage *_prev_message = (prev_message); \ | 
					
						
							|  |  |  |  |   struct msghdr *_msg = (msg); \ | 
					
						
							|  |  |  |  |   const struct msghdr *_prev_msg = (prev_msg); \ | 
					
						
							|  |  |  |  |   GError **_error = (error); \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |   _msg->msg_flags = 0; \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |   /* name */ \ | 
					
						
							|  |  |  |  |   if (_prev_message != NULL && _prev_message->address == _message->address) \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       _msg->msg_name = _prev_msg->msg_name; \ | 
					
						
							|  |  |  |  |       _msg->msg_namelen = _prev_msg->msg_namelen; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |   else if (_message->address != NULL) \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       _msg->msg_namelen = g_socket_address_get_native_size (_message->address); \ | 
					
						
							|  |  |  |  |       _msg->msg_name = g_alloca (_msg->msg_namelen); \ | 
					
						
							|  |  |  |  |       if (!g_socket_address_to_native (_message->address, _msg->msg_name, \ | 
					
						
							|  |  |  |  |                                        _msg->msg_namelen, _error)) \ | 
					
						
							|  |  |  |  |         break; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |   else \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       _msg->msg_name = NULL; \ | 
					
						
							|  |  |  |  |       _msg->msg_namelen = 0; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |   /* iov */ \ | 
					
						
							|  |  |  |  |   { \ | 
					
						
							|  |  |  |  |     /* this entire expression will be evaluated at compile time */ \ | 
					
						
							|  |  |  |  |     if (sizeof *_msg->msg_iov == sizeof *_message->vectors && \ | 
					
						
							|  |  |  |  |         sizeof _msg->msg_iov->iov_base == sizeof _message->vectors->buffer && \ | 
					
						
							|  |  |  |  |         G_STRUCT_OFFSET (struct iovec, iov_base) == \ | 
					
						
							|  |  |  |  |         G_STRUCT_OFFSET (GOutputVector, buffer) && \ | 
					
						
							|  |  |  |  |         sizeof _msg->msg_iov->iov_len == sizeof _message->vectors->size && \ | 
					
						
							|  |  |  |  |         G_STRUCT_OFFSET (struct iovec, iov_len) == \ | 
					
						
							|  |  |  |  |         G_STRUCT_OFFSET (GOutputVector, size)) \ | 
					
						
							|  |  |  |  |       /* ABI is compatible */ \ | 
					
						
							|  |  |  |  |       { \ | 
					
						
							|  |  |  |  |         _msg->msg_iov = (struct iovec *) _message->vectors; \ | 
					
						
							|  |  |  |  |         _msg->msg_iovlen = _message->num_vectors; \ | 
					
						
							|  |  |  |  |       } \ | 
					
						
							|  |  |  |  |     else \ | 
					
						
							|  |  |  |  |       /* ABI is incompatible */ \ | 
					
						
							|  |  |  |  |       { \ | 
					
						
							|  |  |  |  |         gint i; \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |         _msg->msg_iov = g_newa (struct iovec, _message->num_vectors); \ | 
					
						
							|  |  |  |  |         for (i = 0; i < _message->num_vectors; i++) \ | 
					
						
							|  |  |  |  |           { \ | 
					
						
							|  |  |  |  |             _msg->msg_iov[i].iov_base = (void *) _message->vectors[i].buffer; \ | 
					
						
							|  |  |  |  |             _msg->msg_iov[i].iov_len = _message->vectors[i].size; \ | 
					
						
							|  |  |  |  |           } \ | 
					
						
							|  |  |  |  |         _msg->msg_iovlen = _message->num_vectors; \ | 
					
						
							|  |  |  |  |       } \ | 
					
						
							|  |  |  |  |   } \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |   /* control */ \ | 
					
						
							|  |  |  |  |   { \ | 
					
						
							|  |  |  |  |     struct cmsghdr *cmsg; \ | 
					
						
							|  |  |  |  |     gint i; \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |     _msg->msg_controllen = 0; \ | 
					
						
							|  |  |  |  |     for (i = 0; i < _message->num_control_messages; i++) \ | 
					
						
							|  |  |  |  |       _msg->msg_controllen += CMSG_SPACE (g_socket_control_message_get_size (_message->control_messages[i])); \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |     if (_msg->msg_controllen == 0) \ | 
					
						
							|  |  |  |  |       _msg->msg_control = NULL; \ | 
					
						
							|  |  |  |  |     else \ | 
					
						
							|  |  |  |  |       { \ | 
					
						
							|  |  |  |  |         _msg->msg_control = g_alloca (_msg->msg_controllen); \ | 
					
						
							|  |  |  |  |         memset (_msg->msg_control, '\0', _msg->msg_controllen); \ | 
					
						
							|  |  |  |  |       } \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |     cmsg = CMSG_FIRSTHDR (_msg); \ | 
					
						
							|  |  |  |  |     for (i = 0; i < _message->num_control_messages; i++) \ | 
					
						
							|  |  |  |  |       { \ | 
					
						
							|  |  |  |  |         cmsg->cmsg_level = g_socket_control_message_get_level (_message->control_messages[i]); \ | 
					
						
							|  |  |  |  |         cmsg->cmsg_type = g_socket_control_message_get_msg_type (_message->control_messages[i]); \ | 
					
						
							|  |  |  |  |         cmsg->cmsg_len = CMSG_LEN (g_socket_control_message_get_size (_message->control_messages[i])); \ | 
					
						
							|  |  |  |  |         g_socket_control_message_serialize (_message->control_messages[i], \ | 
					
						
							|  |  |  |  |                                             CMSG_DATA (cmsg)); \ | 
					
						
							|  |  |  |  |         cmsg = CMSG_NXTHDR (_msg, cmsg); \ | 
					
						
							|  |  |  |  |       } \ | 
					
						
							|  |  |  |  |     g_assert (cmsg == NULL); \ | 
					
						
							|  |  |  |  |   } \ | 
					
						
							|  |  |  |  | } G_STMT_END | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #define input_message_to_msghdr(message, msg) \
 | 
					
						
							|  |  |  |  | G_STMT_START { \ | 
					
						
							|  |  |  |  |   const GInputMessage  *_message = (message); \ | 
					
						
							|  |  |  |  |   struct msghdr *_msg = (msg); \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |   /* name */ \ | 
					
						
							|  |  |  |  |   if (_message->address) \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       _msg->msg_namelen = sizeof (struct sockaddr_storage); \ | 
					
						
							|  |  |  |  |       _msg->msg_name = g_alloca (_msg->msg_namelen); \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |   else \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       _msg->msg_name = NULL; \ | 
					
						
							|  |  |  |  |       _msg->msg_namelen = 0; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |   /* iov */ \ | 
					
						
							|  |  |  |  |   /* this entire expression will be evaluated at compile time */ \ | 
					
						
							|  |  |  |  |   if (sizeof *_msg->msg_iov == sizeof *_message->vectors && \ | 
					
						
							|  |  |  |  |       sizeof _msg->msg_iov->iov_base == sizeof _message->vectors->buffer && \ | 
					
						
							|  |  |  |  |       G_STRUCT_OFFSET (struct iovec, iov_base) == \ | 
					
						
							|  |  |  |  |       G_STRUCT_OFFSET (GInputVector, buffer) && \ | 
					
						
							|  |  |  |  |       sizeof _msg->msg_iov->iov_len == sizeof _message->vectors->size && \ | 
					
						
							|  |  |  |  |       G_STRUCT_OFFSET (struct iovec, iov_len) == \ | 
					
						
							|  |  |  |  |       G_STRUCT_OFFSET (GInputVector, size)) \ | 
					
						
							|  |  |  |  |     /* ABI is compatible */ \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       _msg->msg_iov = (struct iovec *) _message->vectors; \ | 
					
						
							|  |  |  |  |       _msg->msg_iovlen = _message->num_vectors; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |   else \ | 
					
						
							|  |  |  |  |     /* ABI is incompatible */ \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       guint i; \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |       _msg->msg_iov = g_newa (struct iovec, _message->num_vectors); \ | 
					
						
							|  |  |  |  |       for (i = 0; i < _message->num_vectors; i++) \ | 
					
						
							|  |  |  |  |         { \ | 
					
						
							|  |  |  |  |           _msg->msg_iov[i].iov_base = _message->vectors[i].buffer; \ | 
					
						
							|  |  |  |  |           _msg->msg_iov[i].iov_len = _message->vectors[i].size; \ | 
					
						
							|  |  |  |  |         } \ | 
					
						
							|  |  |  |  |       _msg->msg_iovlen = _message->num_vectors; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |  \ | 
					
						
							|  |  |  |  |   /* control */ \ | 
					
						
							| 
									
										
										
										
											2016-11-16 13:18:53 +01:00
										 |  |  |  |   if (_message->control_messages == NULL) \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  | 	  _msg->msg_controllen = 0; \ | 
					
						
							|  |  |  |  | 	  _msg->msg_control = NULL; \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							|  |  |  |  |   else \ | 
					
						
							|  |  |  |  |     { \ | 
					
						
							|  |  |  |  |       _msg->msg_controllen = 2048; \ | 
					
						
							|  |  |  |  |       _msg->msg_control = g_alloca (_msg->msg_controllen); \ | 
					
						
							|  |  |  |  |     } \ | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |  \ | 
					
						
							|  |  |  |  |   /* flags */ \ | 
					
						
							|  |  |  |  |   _msg->msg_flags = _message->flags; \ | 
					
						
							|  |  |  |  | } G_STMT_END | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | input_message_from_msghdr (const struct msghdr  *msg, | 
					
						
							|  |  |  |  |                            GInputMessage        *message, | 
					
						
							|  |  |  |  |                            GSocket              *socket) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   /* decode address */ | 
					
						
							|  |  |  |  |   if (message->address != NULL) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       *message->address = cache_recv_address (socket, msg->msg_name, | 
					
						
							|  |  |  |  |                                               msg->msg_namelen); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* decode control messages */ | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     GPtrArray *my_messages = NULL; | 
					
						
							|  |  |  |  |     struct cmsghdr *cmsg; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (msg->msg_controllen >= sizeof (struct cmsghdr)) | 
					
						
							|  |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2016-11-16 13:18:53 +01:00
										 |  |  |  |         g_assert (message->control_messages != NULL); | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |         for (cmsg = CMSG_FIRSTHDR (msg); | 
					
						
							|  |  |  |  |              cmsg != NULL; | 
					
						
							|  |  |  |  |              cmsg = CMSG_NXTHDR ((struct msghdr *) msg, cmsg)) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             GSocketControlMessage *control_message; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             control_message = g_socket_control_message_deserialize (cmsg->cmsg_level, | 
					
						
							|  |  |  |  |                                                                     cmsg->cmsg_type, | 
					
						
							|  |  |  |  |                                                                     cmsg->cmsg_len - ((char *)CMSG_DATA (cmsg) - (char *)cmsg), | 
					
						
							|  |  |  |  |                                                                     CMSG_DATA (cmsg)); | 
					
						
							|  |  |  |  |             if (control_message == NULL) | 
					
						
							|  |  |  |  |               /* We've already spewed about the problem in the
 | 
					
						
							|  |  |  |  |                  deserialization code, so just continue */ | 
					
						
							|  |  |  |  |               continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-16 13:18:53 +01:00
										 |  |  |  |             if (my_messages == NULL) | 
					
						
							|  |  |  |  |               my_messages = g_ptr_array_new (); | 
					
						
							|  |  |  |  |             g_ptr_array_add (my_messages, control_message); | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |            } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (message->num_control_messages) | 
					
						
							|  |  |  |  |       *message->num_control_messages = my_messages != NULL ? my_messages->len : 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (message->control_messages) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         if (my_messages == NULL) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             *message->control_messages = NULL; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         else | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             g_ptr_array_add (my_messages, NULL); | 
					
						
							|  |  |  |  |             *message->control_messages = (GSocketControlMessage **) g_ptr_array_free (my_messages, FALSE); | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         g_assert (my_messages == NULL); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   /* capture the flags */ | 
					
						
							|  |  |  |  |   message->flags = msg->msg_flags; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-10-05 16:29:05 +08:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2009-05-18 13:02:11 +02:00
										 |  |  |  |  * g_socket_send_message: | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @address: (nullable): a #GSocketAddress, or %NULL | 
					
						
							| 
									
										
										
										
											2010-12-29 13:51:44 +01:00
										 |  |  |  |  * @vectors: (array length=num_vectors): an array of #GOutputVector structs | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @num_vectors: the number of elements in @vectors, or -1 | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @messages: (array length=num_messages) (nullable): a pointer to an | 
					
						
							| 
									
										
										
										
											2010-12-29 13:51:44 +01:00
										 |  |  |  |  *   array of #GSocketControlMessages, or %NULL. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @num_messages: number of elements in @messages, or -1. | 
					
						
							|  |  |  |  |  * @flags: an int containing #GSocketMsgFlags flags | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-08-17 19:13:15 +01:00
										 |  |  |  |  * Send data to @address on @socket.  For sending multiple messages see | 
					
						
							|  |  |  |  |  * g_socket_send_messages(); for easier use, see | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * g_socket_send() and g_socket_send_to(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If @address is %NULL then the message is sent to the default receiver | 
					
						
							|  |  |  |  |  * (set by g_socket_connect()). | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * @vectors must point to an array of #GOutputVector structs and | 
					
						
							|  |  |  |  |  * @num_vectors must be the length of this array. (If @num_vectors is -1, | 
					
						
							|  |  |  |  |  * then @vectors is assumed to be terminated by a #GOutputVector with a | 
					
						
							|  |  |  |  |  * %NULL buffer pointer.) The #GOutputVector structs describe the buffers | 
					
						
							|  |  |  |  |  * that the sent data will be gathered from. Using multiple | 
					
						
							| 
									
										
										
										
											2014-02-01 11:57:13 -05:00
										 |  |  |  |  * #GOutputVectors is more memory-efficient than manually copying | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * data from multiple sources into a single buffer, and more | 
					
						
							|  |  |  |  |  * network-efficient than making multiple calls to g_socket_send(). | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * @messages, if non-%NULL, is taken to point to an array of @num_messages | 
					
						
							|  |  |  |  |  * #GSocketControlMessage instances. These correspond to the control | 
					
						
							|  |  |  |  |  * messages to be sent on the socket. | 
					
						
							|  |  |  |  |  * If @num_messages is -1 then @messages is treated as a %NULL-terminated | 
					
						
							|  |  |  |  |  * array. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * @flags modify how the message is sent. The commonly available arguments | 
					
						
							|  |  |  |  |  * for this are available in the #GSocketMsgFlags enum, but the | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * values there are the same as the system values, and the flags | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * are passed in as-is, so you can pass in system-specific flags too. | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * If the socket is in blocking mode the call will block until there is | 
					
						
							|  |  |  |  |  * space for the data in the socket queue. If there is no space available | 
					
						
							|  |  |  |  |  * and the socket is in non-blocking mode a %G_IO_ERROR_WOULD_BLOCK error | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * will be returned. To be notified when space is available, wait for the | 
					
						
							|  |  |  |  |  * %G_IO_OUT condition. Note though that you may still receive | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_WOULD_BLOCK from g_socket_send() even if you were previously | 
					
						
							|  |  |  |  |  * notified of a %G_IO_OUT condition. (On Windows in particular, this is | 
					
						
							|  |  |  |  |  * very common due to the way the underlying APIs work.) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * On error -1 is returned and @error is set accordingly. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-05-23 21:49:47 -03:00
										 |  |  |  |  * Returns: Number of bytes written (which may be less than @size), or -1 | 
					
						
							|  |  |  |  |  * on error | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							| 
									
										
										
										
											2009-05-27 18:20:08 -04:00
										 |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | gssize | 
					
						
							|  |  |  |  | g_socket_send_message (GSocket                *socket, | 
					
						
							|  |  |  |  | 		       GSocketAddress         *address, | 
					
						
							|  |  |  |  | 		       GOutputVector          *vectors, | 
					
						
							|  |  |  |  | 		       gint                    num_vectors, | 
					
						
							|  |  |  |  | 		       GSocketControlMessage **messages, | 
					
						
							|  |  |  |  | 		       gint                    num_messages, | 
					
						
							|  |  |  |  | 		       gint                    flags, | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  | 		       GCancellable           *cancellable, | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		       GError                **error) | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   return g_socket_send_message_with_timeout (socket, address, | 
					
						
							|  |  |  |  |                                              vectors, num_vectors, | 
					
						
							|  |  |  |  |                                              messages, num_messages, flags, | 
					
						
							|  |  |  |  |                                              socket->priv->blocking ? -1 : 0, | 
					
						
							|  |  |  |  |                                              cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gssize | 
					
						
							|  |  |  |  | g_socket_send_message_with_timeout (GSocket                *socket, | 
					
						
							|  |  |  |  |                                     GSocketAddress         *address, | 
					
						
							|  |  |  |  |                                     GOutputVector          *vectors, | 
					
						
							|  |  |  |  |                                     gint                    num_vectors, | 
					
						
							|  |  |  |  |                                     GSocketControlMessage **messages, | 
					
						
							|  |  |  |  |                                     gint                    num_messages, | 
					
						
							|  |  |  |  |                                     gint                    flags, | 
					
						
							|  |  |  |  |                                     gint64                  timeout, | 
					
						
							|  |  |  |  |                                     GCancellable           *cancellable, | 
					
						
							|  |  |  |  |                                     GError                **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   GOutputVector one_vector; | 
					
						
							|  |  |  |  |   char zero; | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  |   gint64 start_time; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-04 00:13:50 +02:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), -1); | 
					
						
							| 
									
										
										
										
											2014-05-15 12:16:00 +01:00
										 |  |  |  |   g_return_val_if_fail (address == NULL || G_IS_SOCKET_ADDRESS (address), -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (num_vectors == 0 || vectors != NULL, -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (num_messages == 0 || messages != NULL, -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (cancellable == NULL || G_IS_CANCELLABLE (cancellable), -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (error == NULL || *error == NULL, -1); | 
					
						
							| 
									
										
										
										
											2012-01-04 00:13:50 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  |   start_time = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-14 17:46:38 -04:00
										 |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  |   if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   if (num_vectors == -1) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       for (num_vectors = 0; | 
					
						
							|  |  |  |  | 	   vectors[num_vectors].buffer != NULL; | 
					
						
							|  |  |  |  | 	   num_vectors++) | 
					
						
							|  |  |  |  | 	; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (num_messages == -1) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       for (num_messages = 0; | 
					
						
							|  |  |  |  | 	   messages != NULL && messages[num_messages] != NULL; | 
					
						
							|  |  |  |  | 	   num_messages++) | 
					
						
							|  |  |  |  | 	; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (num_vectors == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       zero = '\0'; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       one_vector.buffer = &zero; | 
					
						
							|  |  |  |  |       one_vector.size = 1; | 
					
						
							|  |  |  |  |       num_vectors = 1; | 
					
						
							|  |  |  |  |       vectors = &one_vector; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   { | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     GOutputMessage output_message; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     struct msghdr msg; | 
					
						
							|  |  |  |  |     gssize result; | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     GError *child_error = NULL; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     output_message.address = address; | 
					
						
							|  |  |  |  |     output_message.vectors = vectors; | 
					
						
							|  |  |  |  |     output_message.num_vectors = num_vectors; | 
					
						
							|  |  |  |  |     output_message.bytes_sent = 0; | 
					
						
							|  |  |  |  |     output_message.control_messages = messages; | 
					
						
							|  |  |  |  |     output_message.num_control_messages = num_messages; | 
					
						
							| 
									
										
										
										
											2010-07-31 23:51:45 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     output_message_to_msghdr (&output_message, NULL, &msg, NULL, &child_error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (child_error != NULL) | 
					
						
							| 
									
										
										
										
											2009-09-10 17:18:13 +01:00
										 |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |         g_propagate_error (error, child_error); | 
					
						
							|  |  |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2009-09-10 17:18:13 +01:00
										 |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     while (1) | 
					
						
							|  |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2009-08-19 12:12:06 -04:00
										 |  |  |  | 	result = sendmsg (socket->priv->fd, &msg, flags | G_SOCKET_DEFAULT_SEND_FLAGS); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	if (result < 0) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	    if (errsv == EINTR) | 
					
						
							|  |  |  |  | 	      continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  | 	    if (timeout != 0 && | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		(errsv == EWOULDBLOCK || | 
					
						
							|  |  |  |  | 		 errsv == EAGAIN)) | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |               { | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  |                 if (!block_on_timeout (socket, G_IO_OUT, timeout, start_time, | 
					
						
							|  |  |  |  |                                        cancellable, error)) | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |                   return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 continue; | 
					
						
							|  |  |  |  |               } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 10:18:06 -04:00
										 |  |  |  | 	    socket_set_error_lazy (error, errsv, _("Error sending message: %s")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	    return -1; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return result; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     struct sockaddr_storage addr; | 
					
						
							|  |  |  |  |     guint addrlen; | 
					
						
							|  |  |  |  |     DWORD bytes_sent; | 
					
						
							|  |  |  |  |     int result; | 
					
						
							|  |  |  |  |     WSABUF *bufs; | 
					
						
							|  |  |  |  |     gint i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Win32 doesn't support control messages.
 | 
					
						
							|  |  |  |  |        Actually this is possible for raw and datagram sockets | 
					
						
							|  |  |  |  |        via WSASendMessage on Vista or later, but that doesn't | 
					
						
							|  |  |  |  |        seem very useful */ | 
					
						
							|  |  |  |  |     if (num_messages != 0) | 
					
						
							|  |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2009-05-19 15:03:14 +02:00
										 |  |  |  |         g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, | 
					
						
							| 
									
										
										
										
											2012-08-16 23:02:41 -04:00
										 |  |  |  |                              _("GSocketControlMessage not supported on Windows")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	return -1; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* iov */ | 
					
						
							|  |  |  |  |     bufs = g_newa (WSABUF, num_vectors); | 
					
						
							|  |  |  |  |     for (i = 0; i < num_vectors; i++) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	bufs[i].buf = (char *)vectors[i].buffer; | 
					
						
							|  |  |  |  | 	bufs[i].len = (gulong)vectors[i].size; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* name */ | 
					
						
							| 
									
										
										
										
											2009-05-20 15:59:11 +02:00
										 |  |  |  |     addrlen = 0; /* Avoid warning */ | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     if (address) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	addrlen = g_socket_address_get_native_size (address); | 
					
						
							| 
									
										
										
										
											2009-05-18 14:10:07 +02:00
										 |  |  |  | 	if (!g_socket_address_to_native (address, &addr, sizeof addr, error)) | 
					
						
							|  |  |  |  | 	  return -1; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     while (1) | 
					
						
							|  |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |         win32_unset_event_mask (socket, FD_WRITE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	if (address) | 
					
						
							|  |  |  |  | 	  result = WSASendTo (socket->priv->fd, | 
					
						
							|  |  |  |  | 			      bufs, num_vectors, | 
					
						
							|  |  |  |  | 			      &bytes_sent, flags, | 
					
						
							|  |  |  |  | 			      (const struct sockaddr *)&addr, addrlen, | 
					
						
							|  |  |  |  | 			      NULL, NULL); | 
					
						
							|  |  |  |  | 	else | 
					
						
							|  |  |  |  | 	  result = WSASend (socket->priv->fd, | 
					
						
							|  |  |  |  | 			    bufs, num_vectors, | 
					
						
							|  |  |  |  | 			    &bytes_sent, flags, | 
					
						
							|  |  |  |  | 			    NULL, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	if (result != 0) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	    if (errsv == WSAEINTR) | 
					
						
							|  |  |  |  | 	      continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	    if (errsv == WSAEWOULDBLOCK) | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |               { | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  |                 if (timeout != 0) | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |                   { | 
					
						
							| 
									
										
										
										
											2015-10-05 10:19:50 +01:00
										 |  |  |  |                     if (!block_on_timeout (socket, G_IO_OUT, timeout, | 
					
						
							|  |  |  |  |                                            start_time, cancellable, error)) | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |                       return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     continue; | 
					
						
							|  |  |  |  |                   } | 
					
						
							|  |  |  |  |               } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 10:18:06 -04:00
										 |  |  |  | 	    socket_set_error_lazy (error, errsv, _("Error sending message: %s")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	    return -1; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return bytes_sent; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_send_messages: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @messages: (array length=num_messages): an array of #GOutputMessage structs | 
					
						
							|  |  |  |  |  * @num_messages: the number of elements in @messages | 
					
						
							|  |  |  |  |  * @flags: an int containing #GSocketMsgFlags flags | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Send multiple data messages from @socket in one go.  This is the most | 
					
						
							|  |  |  |  |  * complicated and fully-featured version of this call. For easier use, see | 
					
						
							|  |  |  |  |  * g_socket_send(), g_socket_send_to(), and g_socket_send_message(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @messages must point to an array of #GOutputMessage structs and | 
					
						
							|  |  |  |  |  * @num_messages must be the length of this array. Each #GOutputMessage | 
					
						
							|  |  |  |  |  * contains an address to send the data to, and a pointer to an array of | 
					
						
							|  |  |  |  |  * #GOutputVector structs to describe the buffers that the data to be sent | 
					
						
							|  |  |  |  |  * for each message will be gathered from. Using multiple #GOutputVectors is | 
					
						
							|  |  |  |  |  * more memory-efficient than manually copying data from multiple sources | 
					
						
							|  |  |  |  |  * into a single buffer, and more network-efficient than making multiple | 
					
						
							|  |  |  |  |  * calls to g_socket_send(). Sending multiple messages in one go avoids the | 
					
						
							|  |  |  |  |  * overhead of making a lot of syscalls in scenarios where a lot of data | 
					
						
							|  |  |  |  |  * packets need to be sent (e.g. high-bandwidth video streaming over RTP/UDP), | 
					
						
							|  |  |  |  |  * or where the same data needs to be sent to multiple recipients. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @flags modify how the message is sent. The commonly available arguments | 
					
						
							|  |  |  |  |  * for this are available in the #GSocketMsgFlags enum, but the | 
					
						
							|  |  |  |  |  * values there are the same as the system values, and the flags | 
					
						
							|  |  |  |  |  * are passed in as-is, so you can pass in system-specific flags too. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If the socket is in blocking mode the call will block until there is | 
					
						
							|  |  |  |  |  * space for all the data in the socket queue. If there is no space available | 
					
						
							|  |  |  |  |  * and the socket is in non-blocking mode a %G_IO_ERROR_WOULD_BLOCK error | 
					
						
							|  |  |  |  |  * will be returned if no data was written at all, otherwise the number of | 
					
						
							|  |  |  |  |  * messages sent will be returned. To be notified when space is available, | 
					
						
							|  |  |  |  |  * wait for the %G_IO_OUT condition. Note though that you may still receive | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_WOULD_BLOCK from g_socket_send() even if you were previously | 
					
						
							|  |  |  |  |  * notified of a %G_IO_OUT condition. (On Windows in particular, this is | 
					
						
							|  |  |  |  |  * very common due to the way the underlying APIs work.) | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-08-17 18:10:43 +01:00
										 |  |  |  |  * On error -1 is returned and @error is set accordingly. An error will only | 
					
						
							|  |  |  |  |  * be returned if zero messages could be sent; otherwise the number of messages | 
					
						
							|  |  |  |  |  * successfully sent before the error will be returned. | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: number of messages sent, or -1 on error. Note that the number of | 
					
						
							|  |  |  |  |  *     messages sent may be smaller than @num_messages if the socket is | 
					
						
							|  |  |  |  |  *     non-blocking or if @num_messages was larger than UIO_MAXIOV (1024), | 
					
						
							|  |  |  |  |  *     in which case the caller may re-try to send the remaining messages. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.44 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gint | 
					
						
							|  |  |  |  | g_socket_send_messages (GSocket        *socket, | 
					
						
							|  |  |  |  | 		        GOutputMessage *messages, | 
					
						
							|  |  |  |  | 		        guint           num_messages, | 
					
						
							|  |  |  |  | 		        gint            flags, | 
					
						
							|  |  |  |  | 		        GCancellable   *cancellable, | 
					
						
							|  |  |  |  | 		        GError        **error) | 
					
						
							| 
									
										
										
										
											2015-06-12 08:32:11 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   return g_socket_send_messages_with_timeout (socket, messages, num_messages, | 
					
						
							|  |  |  |  |                                               flags, | 
					
						
							|  |  |  |  |                                               socket->priv->blocking ? -1 : 0, | 
					
						
							|  |  |  |  |                                               cancellable, error); | 
					
						
							| 
									
										
										
										
											2015-06-12 08:32:11 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gint | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  | g_socket_send_messages_with_timeout (GSocket        *socket, | 
					
						
							|  |  |  |  |                                      GOutputMessage *messages, | 
					
						
							|  |  |  |  |                                      guint           num_messages, | 
					
						
							|  |  |  |  |                                      gint            flags, | 
					
						
							|  |  |  |  |                                      gint64          timeout, | 
					
						
							|  |  |  |  |                                      GCancellable   *cancellable, | 
					
						
							|  |  |  |  |                                      GError        **error) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   gint64 start_time; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (num_messages == 0 || messages != NULL, -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (cancellable == NULL || G_IS_CANCELLABLE (cancellable), -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (error == NULL || *error == NULL, -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   start_time = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (num_messages == 0) | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #if !defined (G_OS_WIN32) && defined (HAVE_SENDMMSG)
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     struct mmsghdr *msgvec; | 
					
						
							| 
									
										
										
										
											2015-08-17 18:10:43 +01:00
										 |  |  |  |     gint i, num_sent; | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef UIO_MAXIOV
 | 
					
						
							|  |  |  |  | #define MAX_NUM_MESSAGES UIO_MAXIOV
 | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  | #define MAX_NUM_MESSAGES 1024
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (num_messages > MAX_NUM_MESSAGES) | 
					
						
							|  |  |  |  |       num_messages = MAX_NUM_MESSAGES; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     msgvec = g_newa (struct mmsghdr, num_messages); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < num_messages; ++i) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         GOutputMessage *msg = &messages[i]; | 
					
						
							|  |  |  |  |         struct msghdr *msg_hdr = &msgvec[i].msg_hdr; | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |         GError *child_error = NULL; | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         msgvec[i].msg_len = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |         output_message_to_msghdr (msg, (i > 0) ? &messages[i - 1] : NULL, | 
					
						
							|  |  |  |  |                                   msg_hdr, (i > 0) ? &msgvec[i - 1].msg_hdr : NULL, | 
					
						
							|  |  |  |  |                                   &child_error); | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |         if (child_error != NULL) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |           { | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |             g_propagate_error (error, child_error); | 
					
						
							|  |  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |           } | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:10:43 +01:00
										 |  |  |  |     for (num_sent = 0; num_sent < num_messages;) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |       { | 
					
						
							|  |  |  |  |         gint ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         ret = sendmmsg (socket->priv->fd, msgvec + num_sent, num_messages - num_sent, | 
					
						
							|  |  |  |  |                         flags | G_SOCKET_DEFAULT_SEND_FLAGS); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (ret < 0) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (errsv == EINTR) | 
					
						
							|  |  |  |  |               continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |             if (timeout != 0 && | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |                 (errsv == EWOULDBLOCK || | 
					
						
							|  |  |  |  |                  errsv == EAGAIN)) | 
					
						
							| 
									
										
										
										
											2015-06-03 13:06:24 +01:00
										 |  |  |  |               { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                 if (!block_on_timeout (socket, G_IO_OUT, timeout, start_time, | 
					
						
							|  |  |  |  |                                        cancellable, error)) | 
					
						
							|  |  |  |  |                   { | 
					
						
							|  |  |  |  |                     if (num_sent > 0) | 
					
						
							|  |  |  |  |                       { | 
					
						
							|  |  |  |  |                         g_clear_error (error); | 
					
						
							|  |  |  |  |                         break; | 
					
						
							|  |  |  |  |                       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     return -1; | 
					
						
							|  |  |  |  |                   } | 
					
						
							| 
									
										
										
										
											2015-06-03 13:06:24 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |                 continue; | 
					
						
							|  |  |  |  |               } | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:10:43 +01:00
										 |  |  |  |             /* If any messages were successfully sent, do not error. */ | 
					
						
							|  |  |  |  |             if (num_sent > 0) | 
					
						
							|  |  |  |  |               break; | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 10:18:06 -04:00
										 |  |  |  |             socket_set_error_lazy (error, errsv, _("Error sending message: %s")); | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:10:43 +01:00
										 |  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |           } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         num_sent += ret; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:10:43 +01:00
										 |  |  |  |     for (i = 0; i < num_sent; ++i) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |       messages[i].bytes_sent = msgvec[i].msg_len; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:10:43 +01:00
										 |  |  |  |     return num_sent; | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     gssize result; | 
					
						
							|  |  |  |  |     gint i; | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |     gint64 wait_timeout; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     wait_timeout = timeout; | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < num_messages; ++i) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         GOutputMessage *msg = &messages[i]; | 
					
						
							|  |  |  |  |         GError *msg_error = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |         result = g_socket_send_message_with_timeout (socket, msg->address, | 
					
						
							|  |  |  |  |                                                      msg->vectors, | 
					
						
							|  |  |  |  |                                                      msg->num_vectors, | 
					
						
							|  |  |  |  |                                                      msg->control_messages, | 
					
						
							|  |  |  |  |                                                      msg->num_control_messages, | 
					
						
							|  |  |  |  |                                                      flags, wait_timeout, | 
					
						
							|  |  |  |  |                                                      cancellable, &msg_error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* check if we've timed out or how much time to wait at most */ | 
					
						
							|  |  |  |  |         if (timeout > 0) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             gint64 elapsed = g_get_monotonic_time () - start_time; | 
					
						
							|  |  |  |  |             wait_timeout = MAX (timeout - elapsed, 1); | 
					
						
							|  |  |  |  |           } | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (result < 0) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             /* if we couldn't send all messages, just return how many we did
 | 
					
						
							|  |  |  |  |              * manage to send, provided we managed to send at least one */ | 
					
						
							| 
									
										
										
										
											2016-07-11 17:46:41 -04:00
										 |  |  |  |             if (i > 0) | 
					
						
							| 
									
										
										
										
											2014-06-12 18:16:45 +01:00
										 |  |  |  |               { | 
					
						
							|  |  |  |  |                 g_error_free (msg_error); | 
					
						
							|  |  |  |  |                 return i; | 
					
						
							|  |  |  |  |               } | 
					
						
							|  |  |  |  |             else | 
					
						
							|  |  |  |  |               { | 
					
						
							|  |  |  |  |                 g_propagate_error (error, msg_error); | 
					
						
							|  |  |  |  |                 return -1; | 
					
						
							|  |  |  |  |               } | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         msg->bytes_sent = result; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return i; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-09 15:28:36 +01:00
										 |  |  |  | static GSocketAddress * | 
					
						
							|  |  |  |  | cache_recv_address (GSocket *socket, struct sockaddr *native, int native_len) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GSocketAddress *saddr; | 
					
						
							|  |  |  |  |   gint i; | 
					
						
							|  |  |  |  |   guint64 oldest_time = G_MAXUINT64; | 
					
						
							|  |  |  |  |   gint oldest_index = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (native_len <= 0) | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saddr = NULL; | 
					
						
							|  |  |  |  |   for (i = 0; i < RECV_ADDR_CACHE_SIZE; i++) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       GSocketAddress *tmp = socket->priv->recv_addr_cache[i].addr; | 
					
						
							|  |  |  |  |       gpointer tmp_native = socket->priv->recv_addr_cache[i].native; | 
					
						
							|  |  |  |  |       gint tmp_native_len = socket->priv->recv_addr_cache[i].native_len; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (!tmp) | 
					
						
							|  |  |  |  |         continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (tmp_native_len != native_len) | 
					
						
							|  |  |  |  |         continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (memcmp (tmp_native, native, native_len) == 0) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           saddr = g_object_ref (tmp); | 
					
						
							|  |  |  |  |           socket->priv->recv_addr_cache[i].last_used = g_get_monotonic_time (); | 
					
						
							|  |  |  |  |           return saddr; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       if (socket->priv->recv_addr_cache[i].last_used < oldest_time) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |           oldest_time = socket->priv->recv_addr_cache[i].last_used; | 
					
						
							|  |  |  |  |           oldest_index = i; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   saddr = g_socket_address_new_from_native (native, native_len); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (socket->priv->recv_addr_cache[oldest_index].addr) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       g_object_unref (socket->priv->recv_addr_cache[oldest_index].addr); | 
					
						
							|  |  |  |  |       g_free (socket->priv->recv_addr_cache[oldest_index].native); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   socket->priv->recv_addr_cache[oldest_index].native = g_memdup (native, native_len); | 
					
						
							|  |  |  |  |   socket->priv->recv_addr_cache[oldest_index].native_len = native_len; | 
					
						
							|  |  |  |  |   socket->priv->recv_addr_cache[oldest_index].addr = g_object_ref (saddr); | 
					
						
							|  |  |  |  |   socket->priv->recv_addr_cache[oldest_index].last_used = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return saddr; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  | static gssize | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  | g_socket_receive_message_with_timeout (GSocket                 *socket, | 
					
						
							|  |  |  |  |                                        GSocketAddress         **address, | 
					
						
							|  |  |  |  |                                        GInputVector            *vectors, | 
					
						
							|  |  |  |  |                                        gint                     num_vectors, | 
					
						
							|  |  |  |  |                                        GSocketControlMessage ***messages, | 
					
						
							|  |  |  |  |                                        gint                    *num_messages, | 
					
						
							|  |  |  |  |                                        gint                    *flags, | 
					
						
							|  |  |  |  |                                        gint64                   timeout, | 
					
						
							|  |  |  |  |                                        GCancellable            *cancellable, | 
					
						
							|  |  |  |  |                                        GError                 **error) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |   GInputVector one_vector; | 
					
						
							|  |  |  |  |   char one_byte; | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   gint64 start_time; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-04 00:13:50 +02:00
										 |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   start_time = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							| 
									
										
										
										
											2014-03-14 17:46:38 -04:00
										 |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-23 17:42:01 -04:00
										 |  |  |  |   if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |   if (num_vectors == -1) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       for (num_vectors = 0; | 
					
						
							|  |  |  |  | 	   vectors[num_vectors].buffer != NULL; | 
					
						
							|  |  |  |  | 	   num_vectors++) | 
					
						
							|  |  |  |  | 	; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (num_vectors == 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       one_vector.buffer = &one_byte; | 
					
						
							|  |  |  |  |       one_vector.size = 1; | 
					
						
							|  |  |  |  |       num_vectors = 1; | 
					
						
							|  |  |  |  |       vectors = &one_vector; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   { | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     GInputMessage input_message; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     struct msghdr msg; | 
					
						
							|  |  |  |  |     gssize result; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     input_message.address = address; | 
					
						
							|  |  |  |  |     input_message.vectors = vectors; | 
					
						
							|  |  |  |  |     input_message.num_vectors = num_vectors; | 
					
						
							|  |  |  |  |     input_message.bytes_received = 0; | 
					
						
							|  |  |  |  |     input_message.flags = (flags != NULL) ? *flags : 0; | 
					
						
							|  |  |  |  |     input_message.control_messages = messages; | 
					
						
							|  |  |  |  |     input_message.num_control_messages = (guint *) num_messages; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-05 13:16:54 -04:00
										 |  |  |  |     /* We always set the close-on-exec flag so we don't leak file
 | 
					
						
							|  |  |  |  |      * descriptors into child processes.  Note that gunixfdmessage.c | 
					
						
							|  |  |  |  |      * will later call fcntl (fd, FD_CLOEXEC), but that isn't atomic. | 
					
						
							|  |  |  |  |      */ | 
					
						
							|  |  |  |  | #ifdef MSG_CMSG_CLOEXEC
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     input_message.flags |= MSG_CMSG_CLOEXEC; | 
					
						
							| 
									
										
										
										
											2011-05-05 13:16:54 -04:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     input_message_to_msghdr (&input_message, &msg); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     /* do it */ | 
					
						
							|  |  |  |  |     while (1) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	result = recvmsg (socket->priv->fd, &msg, msg.msg_flags); | 
					
						
							| 
									
										
										
										
											2011-05-05 13:16:54 -04:00
										 |  |  |  | #ifdef MSG_CMSG_CLOEXEC	
 | 
					
						
							|  |  |  |  | 	if (result < 0 && get_socket_errno () == EINVAL) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    /* We must be running on an old kernel.  Call without the flag. */ | 
					
						
							|  |  |  |  | 	    msg.msg_flags &= ~(MSG_CMSG_CLOEXEC); | 
					
						
							|  |  |  |  | 	    result = recvmsg (socket->priv->fd, &msg, msg.msg_flags); | 
					
						
							|  |  |  |  | 	  } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 	if (result < 0) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	    if (errsv == EINTR) | 
					
						
							|  |  |  |  | 	      continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  | 	    if (timeout != 0 && | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 		(errsv == EWOULDBLOCK || | 
					
						
							|  |  |  |  | 		 errsv == EAGAIN)) | 
					
						
							| 
									
										
										
										
											2015-06-03 13:06:24 +01:00
										 |  |  |  | 	      { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                 if (!block_on_timeout (socket, G_IO_IN, timeout, start_time, | 
					
						
							|  |  |  |  |                                        cancellable, error)) | 
					
						
							|  |  |  |  |                   return -1; | 
					
						
							| 
									
										
										
										
											2015-06-03 13:06:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2015-06-03 13:06:24 +01:00
										 |  |  |  | 	      } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 10:18:06 -04:00
										 |  |  |  | 	    socket_set_error_lazy (error, errsv, _("Error receiving message: %s")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	    return -1; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |     input_message_from_msghdr (&msg, &input_message, socket); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (flags != NULL) | 
					
						
							| 
									
										
										
										
											2015-06-12 08:30:20 +01:00
										 |  |  |  |       *flags = input_message.flags; | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     return result; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     struct sockaddr_storage addr; | 
					
						
							|  |  |  |  |     int addrlen; | 
					
						
							|  |  |  |  |     DWORD bytes_received; | 
					
						
							|  |  |  |  |     DWORD win_flags; | 
					
						
							|  |  |  |  |     int result; | 
					
						
							|  |  |  |  |     WSABUF *bufs; | 
					
						
							|  |  |  |  |     gint i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* iov */ | 
					
						
							|  |  |  |  |     bufs = g_newa (WSABUF, num_vectors); | 
					
						
							|  |  |  |  |     for (i = 0; i < num_vectors; i++) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  | 	bufs[i].buf = (char *)vectors[i].buffer; | 
					
						
							|  |  |  |  | 	bufs[i].len = (gulong)vectors[i].size; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* flags */ | 
					
						
							|  |  |  |  |     if (flags != NULL) | 
					
						
							|  |  |  |  |       win_flags = *flags; | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |       win_flags = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* do it */ | 
					
						
							|  |  |  |  |     while (1) | 
					
						
							|  |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2016-12-05 12:28:57 +02:00
										 |  |  |  |         win32_unset_event_mask (socket, FD_READ); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	addrlen = sizeof addr; | 
					
						
							|  |  |  |  | 	if (address) | 
					
						
							|  |  |  |  | 	  result = WSARecvFrom (socket->priv->fd, | 
					
						
							|  |  |  |  | 				bufs, num_vectors, | 
					
						
							|  |  |  |  | 				&bytes_received, &win_flags, | 
					
						
							|  |  |  |  | 				(struct sockaddr *)&addr, &addrlen, | 
					
						
							|  |  |  |  | 				NULL, NULL); | 
					
						
							|  |  |  |  | 	else | 
					
						
							|  |  |  |  | 	  result = WSARecv (socket->priv->fd, | 
					
						
							|  |  |  |  | 			    bufs, num_vectors, | 
					
						
							|  |  |  |  | 			    &bytes_received, &win_flags, | 
					
						
							|  |  |  |  | 			    NULL, NULL); | 
					
						
							|  |  |  |  | 	if (result != 0) | 
					
						
							|  |  |  |  | 	  { | 
					
						
							|  |  |  |  | 	    int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 	    if (errsv == WSAEINTR) | 
					
						
							|  |  |  |  | 	      continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |             if (errsv == WSAEWOULDBLOCK) | 
					
						
							|  |  |  |  |               { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                 if (timeout != 0) | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |                   { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                     if (!block_on_timeout (socket, G_IO_IN, timeout, | 
					
						
							|  |  |  |  |                                            start_time, cancellable, error)) | 
					
						
							| 
									
										
										
										
											2014-12-22 16:38:50 +01:00
										 |  |  |  |                       return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     continue; | 
					
						
							|  |  |  |  |                   } | 
					
						
							|  |  |  |  |               } | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-25 10:18:06 -04:00
										 |  |  |  | 	    socket_set_error_lazy (error, errsv, _("Error receiving message: %s")); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  | 	    return -1; | 
					
						
							|  |  |  |  | 	  } | 
					
						
							|  |  |  |  | 	break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* decode address */ | 
					
						
							|  |  |  |  |     if (address != NULL) | 
					
						
							|  |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2012-11-10 10:58:19 -05:00
										 |  |  |  |         *address = cache_recv_address (socket, (struct sockaddr *)&addr, addrlen); | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* capture the flags */ | 
					
						
							|  |  |  |  |     if (flags != NULL) | 
					
						
							|  |  |  |  |       *flags = win_flags; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-26 16:37:02 -04:00
										 |  |  |  |     if (messages != NULL) | 
					
						
							|  |  |  |  |       *messages = NULL; | 
					
						
							| 
									
										
										
										
											2010-05-03 19:24:35 +03:00
										 |  |  |  |     if (num_messages != NULL) | 
					
						
							|  |  |  |  |       *num_messages = 0; | 
					
						
							| 
									
										
										
										
											2010-04-26 16:37:02 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-14 15:26:37 +02:00
										 |  |  |  |     return bytes_received; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-12 08:47:37 +01:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_receive_messages: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							|  |  |  |  |  * @messages: (array length=num_messages): an array of #GInputMessage structs | 
					
						
							|  |  |  |  |  * @num_messages: the number of elements in @messages | 
					
						
							|  |  |  |  |  * @flags: an int containing #GSocketMsgFlags flags for the overall operation | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @cancellable: (nullable): a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2015-06-12 08:47:37 +01:00
										 |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Receive multiple data messages from @socket in one go.  This is the most | 
					
						
							|  |  |  |  |  * complicated and fully-featured version of this call. For easier use, see | 
					
						
							|  |  |  |  |  * g_socket_receive(), g_socket_receive_from(), and g_socket_receive_message(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @messages must point to an array of #GInputMessage structs and | 
					
						
							|  |  |  |  |  * @num_messages must be the length of this array. Each #GInputMessage | 
					
						
							|  |  |  |  |  * contains a pointer to an array of #GInputVector structs describing the | 
					
						
							|  |  |  |  |  * buffers that the data received in each message will be written to. Using | 
					
						
							|  |  |  |  |  * multiple #GInputVectors is more memory-efficient than manually copying data | 
					
						
							|  |  |  |  |  * out of a single buffer to multiple sources, and more system-call-efficient | 
					
						
							|  |  |  |  |  * than making multiple calls to g_socket_receive(), such as in scenarios where | 
					
						
							|  |  |  |  |  * a lot of data packets need to be received (e.g. high-bandwidth video | 
					
						
							|  |  |  |  |  * streaming over RTP/UDP). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @flags modify how all messages are received. The commonly available | 
					
						
							|  |  |  |  |  * arguments for this are available in the #GSocketMsgFlags enum, but the | 
					
						
							|  |  |  |  |  * values there are the same as the system values, and the flags | 
					
						
							|  |  |  |  |  * are passed in as-is, so you can pass in system-specific flags too. These | 
					
						
							|  |  |  |  |  * flags affect the overall receive operation. Flags affecting individual | 
					
						
							|  |  |  |  |  * messages are returned in #GInputMessage.flags. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * The other members of #GInputMessage are treated as described in its | 
					
						
							|  |  |  |  |  * documentation. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If #GSocket:blocking is %TRUE the call will block until @num_messages have | 
					
						
							|  |  |  |  |  * been received, or the end of the stream is reached. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If #GSocket:blocking is %FALSE the call will return up to @num_messages | 
					
						
							|  |  |  |  |  * without blocking, or %G_IO_ERROR_WOULD_BLOCK if no messages are queued in the | 
					
						
							|  |  |  |  |  * operating system to be received. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * In blocking mode, if #GSocket:timeout is positive and is reached before any | 
					
						
							|  |  |  |  |  * messages are received, %G_IO_ERROR_TIMED_OUT is returned, otherwise up to | 
					
						
							|  |  |  |  |  * @num_messages are returned. (Note: This is effectively the | 
					
						
							|  |  |  |  |  * behaviour of `MSG_WAITFORONE` with recvmmsg().) | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * To be notified when messages are available, wait for the | 
					
						
							|  |  |  |  |  * %G_IO_IN condition. Note though that you may still receive | 
					
						
							|  |  |  |  |  * %G_IO_ERROR_WOULD_BLOCK from g_socket_receive_messages() even if you were | 
					
						
							|  |  |  |  |  * previously notified of a %G_IO_IN condition. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If the remote peer closes the connection, any messages queued in the | 
					
						
							|  |  |  |  |  * operating system will be returned, and subsequent calls to | 
					
						
							|  |  |  |  |  * g_socket_receive_messages() will return 0 (with no error set). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * On error -1 is returned and @error is set accordingly. An error will only | 
					
						
							|  |  |  |  |  * be returned if zero messages could be received; otherwise the number of | 
					
						
							|  |  |  |  |  * messages successfully received before the error will be returned. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: number of messages received, or -1 on error. Note that the number | 
					
						
							|  |  |  |  |  *     of messages received may be smaller than @num_messages if in non-blocking | 
					
						
							|  |  |  |  |  *     mode, if the peer closed the connection, or if @num_messages | 
					
						
							|  |  |  |  |  *     was larger than `UIO_MAXIOV` (1024), in which case the caller may re-try | 
					
						
							|  |  |  |  |  *     to receive the remaining messages. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.48 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gint | 
					
						
							|  |  |  |  | g_socket_receive_messages (GSocket        *socket, | 
					
						
							|  |  |  |  |                            GInputMessage  *messages, | 
					
						
							|  |  |  |  |                            guint           num_messages, | 
					
						
							|  |  |  |  |                            gint            flags, | 
					
						
							|  |  |  |  |                            GCancellable   *cancellable, | 
					
						
							|  |  |  |  |                            GError        **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error) || | 
					
						
							|  |  |  |  |       !check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return g_socket_receive_messages_with_timeout (socket, messages, num_messages, | 
					
						
							|  |  |  |  |                                                  flags, | 
					
						
							|  |  |  |  |                                                  socket->priv->blocking ? -1 : 0, | 
					
						
							|  |  |  |  |                                                  cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static gint | 
					
						
							|  |  |  |  | g_socket_receive_messages_with_timeout (GSocket        *socket, | 
					
						
							|  |  |  |  |                                         GInputMessage  *messages, | 
					
						
							|  |  |  |  |                                         guint           num_messages, | 
					
						
							|  |  |  |  |                                         gint            flags, | 
					
						
							|  |  |  |  |                                         gint64          timeout, | 
					
						
							|  |  |  |  |                                         GCancellable   *cancellable, | 
					
						
							|  |  |  |  |                                         GError        **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint64 start_time; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (num_messages == 0 || messages != NULL, -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (cancellable == NULL || | 
					
						
							|  |  |  |  |                         G_IS_CANCELLABLE (cancellable), -1); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (error == NULL || *error == NULL, -1); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   start_time = g_get_monotonic_time (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_socket (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!check_timeout (socket, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (g_cancellable_set_error_if_cancelled (cancellable, error)) | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (num_messages == 0) | 
					
						
							|  |  |  |  |     return 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #if !defined (G_OS_WIN32) && defined (HAVE_RECVMMSG)
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     struct mmsghdr *msgvec; | 
					
						
							|  |  |  |  |     guint i, num_received; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef UIO_MAXIOV
 | 
					
						
							|  |  |  |  | #define MAX_NUM_MESSAGES UIO_MAXIOV
 | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  | #define MAX_NUM_MESSAGES 1024
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (num_messages > MAX_NUM_MESSAGES) | 
					
						
							|  |  |  |  |       num_messages = MAX_NUM_MESSAGES; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     msgvec = g_newa (struct mmsghdr, num_messages); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < num_messages; ++i) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         GInputMessage *msg = &messages[i]; | 
					
						
							|  |  |  |  |         struct msghdr *msg_hdr = &msgvec[i].msg_hdr; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         input_message_to_msghdr (msg, msg_hdr); | 
					
						
							|  |  |  |  |         msgvec[i].msg_len = 0; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* We always set the close-on-exec flag so we don't leak file
 | 
					
						
							|  |  |  |  |      * descriptors into child processes.  Note that gunixfdmessage.c | 
					
						
							|  |  |  |  |      * will later call fcntl (fd, FD_CLOEXEC), but that isn't atomic. | 
					
						
							|  |  |  |  |      */ | 
					
						
							|  |  |  |  | #ifdef MSG_CMSG_CLOEXEC
 | 
					
						
							|  |  |  |  |     flags |= MSG_CMSG_CLOEXEC; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (num_received = 0; num_received < num_messages;) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         gint ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* We operate in non-blocking mode and handle the timeout ourselves. */ | 
					
						
							|  |  |  |  |         ret = recvmmsg (socket->priv->fd, | 
					
						
							|  |  |  |  |                         msgvec + num_received, | 
					
						
							|  |  |  |  |                         num_messages - num_received, | 
					
						
							|  |  |  |  |                         flags | G_SOCKET_DEFAULT_SEND_FLAGS, NULL); | 
					
						
							|  |  |  |  | #ifdef MSG_CMSG_CLOEXEC
 | 
					
						
							|  |  |  |  |         if (ret < 0 && get_socket_errno () == EINVAL) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             /* We must be running on an old kernel. Call without the flag. */ | 
					
						
							|  |  |  |  |             flags &= ~(MSG_CMSG_CLOEXEC); | 
					
						
							|  |  |  |  |             ret = recvmmsg (socket->priv->fd, | 
					
						
							|  |  |  |  |                             msgvec + num_received, | 
					
						
							|  |  |  |  |                             num_messages - num_received, | 
					
						
							|  |  |  |  |                             flags | G_SOCKET_DEFAULT_SEND_FLAGS, NULL); | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (ret < 0) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (errsv == EINTR) | 
					
						
							|  |  |  |  |               continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (timeout != 0 && | 
					
						
							|  |  |  |  |                 (errsv == EWOULDBLOCK || | 
					
						
							|  |  |  |  |                  errsv == EAGAIN)) | 
					
						
							|  |  |  |  |               { | 
					
						
							|  |  |  |  |                 if (!block_on_timeout (socket, G_IO_IN, timeout, start_time, | 
					
						
							|  |  |  |  |                                        cancellable, error)) | 
					
						
							|  |  |  |  |                   { | 
					
						
							|  |  |  |  |                     if (num_received > 0) | 
					
						
							|  |  |  |  |                       { | 
					
						
							|  |  |  |  |                         g_clear_error (error); | 
					
						
							|  |  |  |  |                         break; | 
					
						
							|  |  |  |  |                       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     return -1; | 
					
						
							|  |  |  |  |                   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 continue; | 
					
						
							|  |  |  |  |               } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             /* If any messages were successfully received, do not error. */ | 
					
						
							|  |  |  |  |             if (num_received > 0) | 
					
						
							|  |  |  |  |               break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             socket_set_error_lazy (error, errsv, | 
					
						
							|  |  |  |  |                                    _("Error receiving message: %s")); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             return -1; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  |         else if (ret == 0) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             /* EOS. */ | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         num_received += ret; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < num_received; ++i) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         input_message_from_msghdr (&msgvec[i].msg_hdr, &messages[i], socket); | 
					
						
							|  |  |  |  |         messages[i].bytes_received = msgvec[i].msg_len; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return num_received; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     guint i; | 
					
						
							|  |  |  |  |     gint64 wait_timeout; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     wait_timeout = timeout; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < num_messages; i++) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         GInputMessage *msg = &messages[i]; | 
					
						
							|  |  |  |  |         gssize len; | 
					
						
							|  |  |  |  |         GError *msg_error = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         msg->flags = flags;  /* in-out parameter */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         len = g_socket_receive_message_with_timeout (socket, | 
					
						
							|  |  |  |  |                                                      msg->address, | 
					
						
							|  |  |  |  |                                                      msg->vectors, | 
					
						
							|  |  |  |  |                                                      msg->num_vectors, | 
					
						
							|  |  |  |  |                                                      msg->control_messages, | 
					
						
							|  |  |  |  |                                                      (gint *) msg->num_control_messages, | 
					
						
							|  |  |  |  |                                                      &msg->flags, | 
					
						
							|  |  |  |  |                                                      wait_timeout, | 
					
						
							|  |  |  |  |                                                      cancellable, | 
					
						
							|  |  |  |  |                                                      &msg_error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* check if we've timed out or how much time to wait at most */ | 
					
						
							|  |  |  |  |         if (timeout > 0) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             gint64 elapsed = g_get_monotonic_time () - start_time; | 
					
						
							|  |  |  |  |             wait_timeout = MAX (timeout - elapsed, 1); | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (len >= 0) | 
					
						
							|  |  |  |  |           msg->bytes_received = len; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (i != 0 && | 
					
						
							|  |  |  |  |             (g_error_matches (msg_error, G_IO_ERROR, G_IO_ERROR_WOULD_BLOCK) || | 
					
						
							|  |  |  |  |              g_error_matches (msg_error, G_IO_ERROR, G_IO_ERROR_TIMED_OUT))) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             g_clear_error (&msg_error); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (msg_error != NULL) | 
					
						
							|  |  |  |  |           { | 
					
						
							|  |  |  |  |             g_propagate_error (error, msg_error); | 
					
						
							|  |  |  |  |             return -1; | 
					
						
							|  |  |  |  |           } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (len == 0) | 
					
						
							|  |  |  |  |           break; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return i; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_receive_message: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @address: (out) (optional): a pointer to a #GSocketAddress | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |  *     pointer, or %NULL | 
					
						
							|  |  |  |  |  * @vectors: (array length=num_vectors): an array of #GInputVector structs | 
					
						
							|  |  |  |  |  * @num_vectors: the number of elements in @vectors, or -1 | 
					
						
							| 
									
										
										
										
											2016-10-28 18:29:02 -07:00
										 |  |  |  |  * @messages: (array length=num_messages) (out) (optional): a pointer which | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |  *    may be filled with an array of #GSocketControlMessages, or %NULL | 
					
						
							| 
									
										
										
										
											2015-11-23 11:16:29 +01:00
										 |  |  |  |  * @num_messages: (out): a pointer which will be filled with the number of | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |  *    elements in @messages, or %NULL | 
					
						
							| 
									
										
										
										
											2015-11-23 11:16:29 +01:00
										 |  |  |  |  * @flags: (inout): a pointer to an int containing #GSocketMsgFlags flags | 
					
						
							|  |  |  |  |  * @cancellable: a %GCancellable or %NULL | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |  * @error: a #GError pointer, or %NULL | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-06-12 08:47:37 +01:00
										 |  |  |  |  * Receive data from a socket.  For receiving multiple messages, see | 
					
						
							|  |  |  |  |  * g_socket_receive_messages(); for easier use, see | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |  * g_socket_receive() and g_socket_receive_from(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If @address is non-%NULL then @address will be set equal to the | 
					
						
							|  |  |  |  |  * source address of the received packet. | 
					
						
							|  |  |  |  |  * @address is owned by the caller. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @vector must point to an array of #GInputVector structs and | 
					
						
							|  |  |  |  |  * @num_vectors must be the length of this array.  These structs | 
					
						
							|  |  |  |  |  * describe the buffers that received data will be scattered into. | 
					
						
							|  |  |  |  |  * If @num_vectors is -1, then @vectors is assumed to be terminated | 
					
						
							|  |  |  |  |  * by a #GInputVector with a %NULL buffer pointer. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * As a special case, if @num_vectors is 0 (in which case, @vectors | 
					
						
							|  |  |  |  |  * may of course be %NULL), then a single byte is received and | 
					
						
							|  |  |  |  |  * discarded. This is to facilitate the common practice of sending a | 
					
						
							|  |  |  |  |  * single '\0' byte for the purposes of transferring ancillary data. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @messages, if non-%NULL, will be set to point to a newly-allocated | 
					
						
							|  |  |  |  |  * array of #GSocketControlMessage instances or %NULL if no such | 
					
						
							|  |  |  |  |  * messages was received. These correspond to the control messages | 
					
						
							|  |  |  |  |  * received from the kernel, one #GSocketControlMessage per message | 
					
						
							|  |  |  |  |  * from the kernel. This array is %NULL-terminated and must be freed | 
					
						
							|  |  |  |  |  * by the caller using g_free() after calling g_object_unref() on each | 
					
						
							|  |  |  |  |  * element. If @messages is %NULL, any control messages received will | 
					
						
							|  |  |  |  |  * be discarded. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @num_messages, if non-%NULL, will be set to the number of control | 
					
						
							|  |  |  |  |  * messages received. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If both @messages and @num_messages are non-%NULL, then | 
					
						
							|  |  |  |  |  * @num_messages gives the number of #GSocketControlMessage instances | 
					
						
							|  |  |  |  |  * in @messages (ie: not including the %NULL terminator). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @flags is an in/out parameter. The commonly available arguments | 
					
						
							|  |  |  |  |  * for this are available in the #GSocketMsgFlags enum, but the | 
					
						
							|  |  |  |  |  * values there are the same as the system values, and the flags | 
					
						
							|  |  |  |  |  * are passed in as-is, so you can pass in system-specific flags too | 
					
						
							|  |  |  |  |  * (and g_socket_receive_message() may pass system-specific flags out). | 
					
						
							|  |  |  |  |  * Flags passed in to the parameter affect the receive operation; flags returned | 
					
						
							|  |  |  |  |  * out of it are relevant to the specific returned message. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * As with g_socket_receive(), data may be discarded if @socket is | 
					
						
							|  |  |  |  |  * %G_SOCKET_TYPE_DATAGRAM or %G_SOCKET_TYPE_SEQPACKET and you do not | 
					
						
							|  |  |  |  |  * provide enough buffer space to read a complete message. You can pass | 
					
						
							|  |  |  |  |  * %G_SOCKET_MSG_PEEK in @flags to peek at the current message without | 
					
						
							|  |  |  |  |  * removing it from the receive queue, but there is no portable way to find | 
					
						
							|  |  |  |  |  * out the length of the message other than by reading it into a | 
					
						
							|  |  |  |  |  * sufficiently-large buffer. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If the socket is in blocking mode the call will block until there | 
					
						
							|  |  |  |  |  * is some data to receive, the connection is closed, or there is an | 
					
						
							|  |  |  |  |  * error. If there is no data available and the socket is in | 
					
						
							|  |  |  |  |  * non-blocking mode, a %G_IO_ERROR_WOULD_BLOCK error will be | 
					
						
							|  |  |  |  |  * returned. To be notified when data is available, wait for the | 
					
						
							|  |  |  |  |  * %G_IO_IN condition. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * On error -1 is returned and @error is set accordingly. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: Number of bytes read, or 0 if the connection was closed by | 
					
						
							|  |  |  |  |  * the peer, or -1 on error | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.22 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gssize | 
					
						
							|  |  |  |  | g_socket_receive_message (GSocket                 *socket, | 
					
						
							|  |  |  |  | 			  GSocketAddress         **address, | 
					
						
							|  |  |  |  | 			  GInputVector            *vectors, | 
					
						
							|  |  |  |  | 			  gint                     num_vectors, | 
					
						
							|  |  |  |  | 			  GSocketControlMessage ***messages, | 
					
						
							|  |  |  |  | 			  gint                    *num_messages, | 
					
						
							|  |  |  |  | 			  gint                    *flags, | 
					
						
							|  |  |  |  | 			  GCancellable            *cancellable, | 
					
						
							|  |  |  |  | 			  GError                 **error) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |   return g_socket_receive_message_with_timeout (socket, address, vectors, | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |                                                  num_vectors, messages, | 
					
						
							|  |  |  |  |                                                  num_messages, flags, | 
					
						
							| 
									
										
										
										
											2015-07-29 11:13:33 +01:00
										 |  |  |  |                                                  socket->priv->blocking ? -1 : 0, | 
					
						
							| 
									
										
										
										
											2015-07-27 14:21:00 +01:00
										 |  |  |  |                                                  cancellable, error); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_credentials: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket. | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns the credentials of the foreign process connected to this | 
					
						
							|  |  |  |  |  * socket, if any (e.g. it is only supported for %G_SOCKET_FAMILY_UNIX | 
					
						
							|  |  |  |  |  * sockets). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If this operation isn't supported on the OS, the method fails with | 
					
						
							|  |  |  |  |  * the %G_IO_ERROR_NOT_SUPPORTED error. On Linux this is implemented | 
					
						
							|  |  |  |  |  * by reading the %SO_PEERCRED option on the underlying socket. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Other ways to obtain credentials from a foreign peer includes the | 
					
						
							|  |  |  |  |  * #GUnixCredentialsMessage type and | 
					
						
							|  |  |  |  |  * g_unix_connection_send_credentials() / | 
					
						
							|  |  |  |  |  * g_unix_connection_receive_credentials() functions. | 
					
						
							|  |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-09-24 18:24:41 -03:00
										 |  |  |  |  * Returns: (transfer full): %NULL if @error is set, otherwise a #GCredentials object | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  |  * that must be freed with g_object_unref(). | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.26 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | GCredentials * | 
					
						
							|  |  |  |  | g_socket_get_credentials (GSocket   *socket, | 
					
						
							|  |  |  |  |                           GError   **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   GCredentials *ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), NULL); | 
					
						
							|  |  |  |  |   g_return_val_if_fail (error == NULL || *error == NULL, NULL); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   ret = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-18 13:40:09 -04:00
										 |  |  |  | #if G_CREDENTIALS_SOCKET_GET_CREDENTIALS_SUPPORTED
 | 
					
						
							| 
									
										
										
										
											2013-09-19 16:09:38 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #ifdef SO_PEERCRED
 | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  |   { | 
					
						
							| 
									
										
										
										
											2013-09-18 13:40:09 -04:00
										 |  |  |  |     guint8 native_creds_buf[G_CREDENTIALS_NATIVE_SIZE]; | 
					
						
							|  |  |  |  |     socklen_t optlen = sizeof (native_creds_buf); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  |     if (getsockopt (socket->priv->fd, | 
					
						
							|  |  |  |  |                     SOL_SOCKET, | 
					
						
							|  |  |  |  |                     SO_PEERCRED, | 
					
						
							| 
									
										
										
										
											2013-09-18 13:40:09 -04:00
										 |  |  |  |                     native_creds_buf, | 
					
						
							|  |  |  |  |                     &optlen) == 0) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         ret = g_credentials_new (); | 
					
						
							|  |  |  |  |         g_credentials_set_native (ret, | 
					
						
							|  |  |  |  |                                   G_CREDENTIALS_NATIVE_TYPE, | 
					
						
							|  |  |  |  |                                   native_creds_buf); | 
					
						
							|  |  |  |  |       } | 
					
						
							| 
									
										
										
										
											2013-09-19 16:09:38 -04:00
										 |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-04-15 15:09:22 +01:00
										 |  |  |  | #elif G_CREDENTIALS_USE_NETBSD_UNPCBID
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     struct unpcbid cred; | 
					
						
							|  |  |  |  |     socklen_t optlen = sizeof (cred); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (getsockopt (socket->priv->fd, | 
					
						
							|  |  |  |  |                     0, | 
					
						
							|  |  |  |  |                     LOCAL_PEEREID, | 
					
						
							|  |  |  |  |                     &cred, | 
					
						
							|  |  |  |  |                     &optlen) == 0) | 
					
						
							|  |  |  |  |       { | 
					
						
							|  |  |  |  |         ret = g_credentials_new (); | 
					
						
							|  |  |  |  |         g_credentials_set_native (ret, | 
					
						
							|  |  |  |  |                                   G_CREDENTIALS_NATIVE_TYPE, | 
					
						
							|  |  |  |  |                                   &cred); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-09-19 16:09:38 -04:00
										 |  |  |  | #elif G_CREDENTIALS_USE_SOLARIS_UCRED
 | 
					
						
							|  |  |  |  |   { | 
					
						
							|  |  |  |  |     ucred_t *ucred = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (getpeerucred (socket->priv->fd, &ucred) == 0) | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  |       { | 
					
						
							| 
									
										
										
										
											2013-09-19 16:09:38 -04:00
										 |  |  |  |         ret = g_credentials_new (); | 
					
						
							|  |  |  |  |         g_credentials_set_native (ret, | 
					
						
							|  |  |  |  |                                   G_CREDENTIALS_TYPE_SOLARIS_UCRED, | 
					
						
							|  |  |  |  |                                   ucred); | 
					
						
							|  |  |  |  |         ucred_free (ucred); | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  |       } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2013-09-19 16:09:38 -04:00
										 |  |  |  |   #error "G_CREDENTIALS_SOCKET_GET_CREDENTIALS_SUPPORTED is set but this is no code for this platform"
 | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (!ret) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       g_set_error (error, | 
					
						
							|  |  |  |  |                    G_IO_ERROR, | 
					
						
							|  |  |  |  |                    socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  |                    _("Unable to read socket credentials: %s"), | 
					
						
							|  |  |  |  |                    socket_strerror (errsv)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #else
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-20 14:02:14 -04:00
										 |  |  |  |   g_set_error_literal (error, | 
					
						
							|  |  |  |  |                        G_IO_ERROR, | 
					
						
							|  |  |  |  |                        G_IO_ERROR_NOT_SUPPORTED, | 
					
						
							|  |  |  |  |                        _("g_socket_get_credentials not implemented for this OS")); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_get_option: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2014-02-06 08:04:52 -05:00
										 |  |  |  |  * @level: the "API level" of the option (eg, `SOL_SOCKET`) | 
					
						
							|  |  |  |  |  * @optname: the "name" of the option (eg, `SO_BROADCAST`) | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * @value: (out): return location for the option value | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Gets the value of an integer-valued option on @socket, as with | 
					
						
							| 
									
										
										
										
											2014-02-06 08:04:52 -05:00
										 |  |  |  |  * getsockopt(). (If you need to fetch a  non-integer-valued option, | 
					
						
							|  |  |  |  |  * you will need to call getsockopt() directly.) | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-02-09 02:07:26 -05:00
										 |  |  |  |  * The [<gio/gnetworking.h>][gio-gnetworking.h] | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * header pulls in system headers that will define most of the | 
					
						
							|  |  |  |  |  * standard/portable socket options. For unusual socket protocols or | 
					
						
							|  |  |  |  |  * platform-dependent options, you may need to include additional | 
					
						
							|  |  |  |  |  * headers. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Note that even for socket options that are a single byte in size, | 
					
						
							|  |  |  |  |  * @value is still a pointer to a #gint variable, not a #guchar; | 
					
						
							|  |  |  |  |  * g_socket_get_option() will handle the conversion internally. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: success or failure. On failure, @error will be set, and | 
					
						
							| 
									
										
										
										
											2014-02-06 08:04:52 -05:00
										 |  |  |  |  *   the system error value (`errno` or WSAGetLastError()) will still | 
					
						
							|  |  |  |  |  *   be set to the result of the getsockopt() call. | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.36 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_get_option (GSocket  *socket, | 
					
						
							|  |  |  |  | 		     gint      level, | 
					
						
							|  |  |  |  | 		     gint      optname, | 
					
						
							|  |  |  |  | 		     gint     *value, | 
					
						
							|  |  |  |  | 		     GError  **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   guint size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   *value = 0; | 
					
						
							|  |  |  |  |   size = sizeof (gint); | 
					
						
							|  |  |  |  |   if (getsockopt (socket->priv->fd, level, optname, value, &size) != 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |       int errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       g_set_error_literal (error, | 
					
						
							|  |  |  |  | 			   G_IO_ERROR, | 
					
						
							|  |  |  |  | 			   socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  | 			   socket_strerror (errsv)); | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |       /* Reset errno in case the caller wants to look at it */ | 
					
						
							|  |  |  |  |       errno = errsv; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |       return FALSE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #if G_BYTE_ORDER == G_BIG_ENDIAN
 | 
					
						
							|  |  |  |  |   /* If the returned value is smaller than an int then we need to
 | 
					
						
							|  |  |  |  |    * slide it over into the low-order bytes of *value. | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   if (size != sizeof (gint)) | 
					
						
							|  |  |  |  |     *value = *value >> (8 * (sizeof (gint) - size)); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return TRUE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /**
 | 
					
						
							|  |  |  |  |  * g_socket_set_option: | 
					
						
							|  |  |  |  |  * @socket: a #GSocket | 
					
						
							| 
									
										
										
										
											2014-02-06 08:04:52 -05:00
										 |  |  |  |  * @level: the "API level" of the option (eg, `SOL_SOCKET`) | 
					
						
							|  |  |  |  |  * @optname: the "name" of the option (eg, `SO_BROADCAST`) | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * @value: the value to set the option to | 
					
						
							|  |  |  |  |  * @error: #GError for error reporting, or %NULL to ignore. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Sets the value of an integer-valued option on @socket, as with | 
					
						
							| 
									
										
										
										
											2014-02-06 08:04:52 -05:00
										 |  |  |  |  * setsockopt(). (If you need to set a non-integer-valued option, | 
					
						
							|  |  |  |  |  * you will need to call setsockopt() directly.) | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-02-09 02:07:26 -05:00
										 |  |  |  |  * The [<gio/gnetworking.h>][gio-gnetworking.h] | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * header pulls in system headers that will define most of the | 
					
						
							|  |  |  |  |  * standard/portable socket options. For unusual socket protocols or | 
					
						
							|  |  |  |  |  * platform-dependent options, you may need to include additional | 
					
						
							|  |  |  |  |  * headers. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Returns: success or failure. On failure, @error will be set, and | 
					
						
							| 
									
										
										
										
											2014-02-06 08:04:52 -05:00
										 |  |  |  |  *   the system error value (`errno` or WSAGetLastError()) will still | 
					
						
							|  |  |  |  |  *   be set to the result of the setsockopt() call. | 
					
						
							| 
									
										
										
										
											2012-02-13 21:12:34 -05:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * Since: 2.36 | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | gboolean | 
					
						
							|  |  |  |  | g_socket_set_option (GSocket  *socket, | 
					
						
							|  |  |  |  | 		     gint      level, | 
					
						
							|  |  |  |  | 		     gint      optname, | 
					
						
							|  |  |  |  | 		     gint      value, | 
					
						
							|  |  |  |  | 		     GError  **error) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |   gint errsv; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (setsockopt (socket->priv->fd, level, optname, &value, sizeof (gint)) == 0) | 
					
						
							|  |  |  |  |     return TRUE; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #if !defined (__linux__) && !defined (G_OS_WIN32)
 | 
					
						
							|  |  |  |  |   /* Linux and Windows let you set a single-byte value from an int,
 | 
					
						
							|  |  |  |  |    * but most other platforms don't. | 
					
						
							|  |  |  |  |    */ | 
					
						
							|  |  |  |  |   if (errno == EINVAL && value >= SCHAR_MIN && value <= CHAR_MAX) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  | #if G_BYTE_ORDER == G_BIG_ENDIAN
 | 
					
						
							|  |  |  |  |       value = value << (8 * (sizeof (gint) - 1)); | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |       if (setsockopt (socket->priv->fd, level, optname, &value, 1) == 0) | 
					
						
							|  |  |  |  |         return TRUE; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   errsv = get_socket_errno (); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   g_set_error_literal (error, | 
					
						
							|  |  |  |  |                        G_IO_ERROR, | 
					
						
							|  |  |  |  |                        socket_io_error_from_errno (errsv), | 
					
						
							|  |  |  |  |                        socket_strerror (errsv)); | 
					
						
							|  |  |  |  | #ifndef G_OS_WIN32
 | 
					
						
							|  |  |  |  |   errno = errsv; | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							|  |  |  |  |   return FALSE; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 |