• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* GStreamer
2  * Copyright (C) 2005 Andy Wingo <wingo@pobox.com>
3  * Copyright (C) 2010 Tim-Philipp Müller <tim centricular net>
4  * Copyright (C) 2012 Collabora Ltd. <tim.muller@collabora.co.uk>
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public
17  * License along with this library; if not, write to the
18  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21 /**
22  * SECTION:gstnettimepacket
23  * @title: GstNetTimePacket
24  * @short_description: Helper structure to construct clock packets used
25  *                     by network clocks.
26  * @see_also: #GstClock, #GstNetClientClock, #GstNetTimeProvider
27  *
28  * Various functions for receiving, sending an serializing #GstNetTimePacket
29  * structures.
30  */
31 
32 #ifdef HAVE_CONFIG_H
33 #include "config.h"
34 #endif
35 
36 #include <glib.h>
37 
38 #ifdef __CYGWIN__
39 # include <unistd.h>
40 # include <fcntl.h>
41 #endif
42 
43 #include "gstnettimepacket.h"
44 
45 G_DEFINE_BOXED_TYPE (GstNetTimePacket, gst_net_time_packet,
46     gst_net_time_packet_copy, gst_net_time_packet_free);
47 
48 /**
49  * gst_net_time_packet_new:
50  * @buffer: (array): a buffer from which to construct the packet, or NULL
51  *
52  * Creates a new #GstNetTimePacket from a buffer received over the network. The
53  * caller is responsible for ensuring that @buffer is at least
54  * #GST_NET_TIME_PACKET_SIZE bytes long.
55  *
56  * If @buffer is %NULL, the local and remote times will be set to
57  * #GST_CLOCK_TIME_NONE.
58  *
59  * MT safe. Caller owns return value (gst_net_time_packet_free to free).
60  *
61  * Returns: The new #GstNetTimePacket.
62  */
63 GstNetTimePacket *
gst_net_time_packet_new(const guint8 * buffer)64 gst_net_time_packet_new (const guint8 * buffer)
65 {
66   GstNetTimePacket *ret;
67 
68   g_assert (sizeof (GstClockTime) == 8);
69 
70   ret = g_new0 (GstNetTimePacket, 1);
71 
72   if (buffer) {
73     ret->local_time = GST_READ_UINT64_BE (buffer);
74     ret->remote_time = GST_READ_UINT64_BE (buffer + sizeof (GstClockTime));
75   } else {
76     ret->local_time = GST_CLOCK_TIME_NONE;
77     ret->remote_time = GST_CLOCK_TIME_NONE;
78   }
79 
80   return ret;
81 }
82 
83 /**
84  * gst_net_time_packet_free:
85  * @packet: the #GstNetTimePacket
86  *
87  * Free @packet.
88  */
89 void
gst_net_time_packet_free(GstNetTimePacket * packet)90 gst_net_time_packet_free (GstNetTimePacket * packet)
91 {
92   g_free (packet);
93 }
94 
95 /**
96  * gst_net_time_packet_copy:
97  * @packet: the #GstNetTimePacket
98  *
99  * Make a copy of @packet.
100  *
101  * Returns: a copy of @packet, free with gst_net_time_packet_free().
102  */
103 GstNetTimePacket *
gst_net_time_packet_copy(const GstNetTimePacket * packet)104 gst_net_time_packet_copy (const GstNetTimePacket * packet)
105 {
106   GstNetTimePacket *ret;
107 
108   ret = g_new0 (GstNetTimePacket, 1);
109   ret->local_time = packet->local_time;
110   ret->remote_time = packet->remote_time;
111 
112   return ret;
113 }
114 
115 /**
116  * gst_net_time_packet_serialize:
117  * @packet: the #GstNetTimePacket
118  *
119  * Serialized a #GstNetTimePacket into a newly-allocated sequence of
120  * #GST_NET_TIME_PACKET_SIZE bytes, in network byte order. The value returned is
121  * suitable for passing to write(2) or sendto(2) for communication over the
122  * network.
123  *
124  * MT safe. Caller owns return value (g_free to free).
125  *
126  * Returns: A newly allocated sequence of #GST_NET_TIME_PACKET_SIZE bytes.
127  */
128 guint8 *
gst_net_time_packet_serialize(const GstNetTimePacket * packet)129 gst_net_time_packet_serialize (const GstNetTimePacket * packet)
130 {
131   guint8 *ret;
132 
133   g_assert (sizeof (GstClockTime) == 8);
134 
135   ret = g_new0 (guint8, GST_NET_TIME_PACKET_SIZE);
136 
137   GST_WRITE_UINT64_BE (ret, packet->local_time);
138   GST_WRITE_UINT64_BE (ret + sizeof (GstClockTime), packet->remote_time);
139 
140   return ret;
141 }
142 
143 /**
144  * gst_net_time_packet_receive:
145  * @socket: socket to receive the time packet on
146  * @src_address: (out): address of variable to return sender address
147  * @error: return address for a #GError, or NULL
148  *
149  * Receives a #GstNetTimePacket over a socket. Handles interrupted system
150  * calls, but otherwise returns NULL on error.
151  *
152  * Returns: (transfer full): a new #GstNetTimePacket, or NULL on error. Free
153  *    with gst_net_time_packet_free() when done.
154  */
155 GstNetTimePacket *
gst_net_time_packet_receive(GSocket * socket,GSocketAddress ** src_address,GError ** error)156 gst_net_time_packet_receive (GSocket * socket,
157     GSocketAddress ** src_address, GError ** error)
158 {
159   gchar buffer[GST_NET_TIME_PACKET_SIZE];
160   GError *err = NULL;
161   gssize ret;
162 
163   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE);
164   g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
165 
166   while (TRUE) {
167     ret = g_socket_receive_from (socket, src_address, buffer,
168         GST_NET_TIME_PACKET_SIZE, NULL, &err);
169 
170     if (ret < 0) {
171       if (err->code == G_IO_ERROR_WOULD_BLOCK) {
172         g_error_free (err);
173         err = NULL;
174         continue;
175       } else {
176         goto receive_error;
177       }
178     } else if (ret < GST_NET_TIME_PACKET_SIZE) {
179       goto short_packet;
180     } else {
181       return gst_net_time_packet_new ((const guint8 *) buffer);
182     }
183   }
184 
185 receive_error:
186   {
187     GST_DEBUG ("receive error: %s", err->message);
188     g_propagate_error (error, err);
189     return NULL;
190   }
191 short_packet:
192   {
193     GST_DEBUG ("someone sent us a short packet (%" G_GSSIZE_FORMAT " < %d)",
194         ret, GST_NET_TIME_PACKET_SIZE);
195     g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_DATA,
196         "short time packet (%d < %d)", (int) ret, GST_NET_TIME_PACKET_SIZE);
197     return NULL;
198   }
199 }
200 
201 /**
202  * gst_net_time_packet_send:
203  * @packet: the #GstNetTimePacket to send
204  * @socket: socket to send the time packet on
205  * @dest_address: address to send the time packet to
206  * @error: return address for a #GError, or NULL
207  *
208  * Sends a #GstNetTimePacket over a socket.
209  *
210  * MT safe.
211  *
212  * Returns: TRUE if successful, FALSE in case an error occurred.
213  */
214 gboolean
gst_net_time_packet_send(const GstNetTimePacket * packet,GSocket * socket,GSocketAddress * dest_address,GError ** error)215 gst_net_time_packet_send (const GstNetTimePacket * packet,
216     GSocket * socket, GSocketAddress * dest_address, GError ** error)
217 {
218   gboolean was_blocking;
219   guint8 *buffer;
220   gssize res;
221 
222   g_return_val_if_fail (packet != NULL, FALSE);
223   g_return_val_if_fail (G_IS_SOCKET (socket), FALSE);
224   g_return_val_if_fail (G_IS_SOCKET_ADDRESS (dest_address), FALSE);
225   g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
226 
227   was_blocking = g_socket_get_blocking (socket);
228 
229   if (was_blocking)
230     g_socket_set_blocking (socket, FALSE);
231 
232   /* FIXME: avoid pointless alloc/free, serialise into stack-allocated buffer */
233   buffer = gst_net_time_packet_serialize (packet);
234 
235   res = g_socket_send_to (socket, dest_address, (const gchar *) buffer,
236       GST_NET_TIME_PACKET_SIZE, NULL, error);
237 
238   /* datagram packets should be sent as a whole or not at all */
239   g_assert (res < 0 || res == GST_NET_TIME_PACKET_SIZE);
240 
241   g_free (buffer);
242 
243   if (was_blocking)
244     g_socket_set_blocking (socket, TRUE);
245 
246   return (res == GST_NET_TIME_PACKET_SIZE);
247 }
248