Coverage Report

Created: 2025-06-13 06:55

/src/glib/gio/gpollableinputstream.c
Line
Count
Source (jump to first uncovered line)
1
/* GIO - GLib Input, Output and Streaming Library
2
 *
3
 * Copyright (C) 2010 Red Hat, Inc.
4
 *
5
 * SPDX-License-Identifier: LGPL-2.1-or-later
6
 *
7
 * This library is free software; you can redistribute it and/or
8
 * modify it under the terms of the GNU Lesser General Public
9
 * License as published by the Free Software Foundation; either
10
 * version 2.1 of the License, or (at your option) any later version.
11
 *
12
 * This library is distributed in the hope that it will be useful,
13
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15
 * Lesser General Public License for more details.
16
 *
17
 * You should have received a copy of the GNU Lesser General
18
 * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
19
 */
20
21
#include "config.h"
22
23
#include <errno.h>
24
25
#include "gpollableinputstream.h"
26
#include "gasynchelper.h"
27
#include "glibintl.h"
28
29
/**
30
 * SECTION:gpollableinputstream
31
 * @short_description: Interface for pollable input streams
32
 * @include: gio/gio.h
33
 * @see_also: #GInputStream, #GPollableOutputStream, #GFileDescriptorBased
34
 *
35
 * #GPollableInputStream is implemented by #GInputStreams that
36
 * can be polled for readiness to read. This can be used when
37
 * interfacing with a non-GIO API that expects
38
 * UNIX-file-descriptor-style asynchronous I/O rather than GIO-style.
39
 *
40
 * Some classes may implement #GPollableInputStream but have only certain
41
 * instances of that class be pollable. If g_pollable_input_stream_can_poll()
42
 * returns %FALSE, then the behavior of other #GPollableInputStream methods is
43
 * undefined.
44
 *
45
 * Since: 2.28
46
 */
47
48
G_DEFINE_INTERFACE (GPollableInputStream, g_pollable_input_stream, G_TYPE_INPUT_STREAM)
49
50
static gboolean g_pollable_input_stream_default_can_poll         (GPollableInputStream *stream);
51
static gssize   g_pollable_input_stream_default_read_nonblocking (GPollableInputStream  *stream,
52
                  void                  *buffer,
53
                  gsize                  count,
54
                  GError               **error);
55
56
static void
57
g_pollable_input_stream_default_init (GPollableInputStreamInterface *iface)
58
0
{
59
0
  iface->can_poll         = g_pollable_input_stream_default_can_poll;
60
0
  iface->read_nonblocking = g_pollable_input_stream_default_read_nonblocking;
61
0
}
62
63
static gboolean
64
g_pollable_input_stream_default_can_poll (GPollableInputStream *stream)
65
0
{
66
0
  return TRUE;
67
0
}
68
69
/**
70
 * g_pollable_input_stream_can_poll:
71
 * @stream: a #GPollableInputStream.
72
 *
73
 * Checks if @stream is actually pollable. Some classes may implement
74
 * #GPollableInputStream but have only certain instances of that class
75
 * be pollable. If this method returns %FALSE, then the behavior of
76
 * other #GPollableInputStream methods is undefined.
77
 *
78
 * For any given stream, the value returned by this method is constant;
79
 * a stream cannot switch from pollable to non-pollable or vice versa.
80
 *
81
 * Returns: %TRUE if @stream is pollable, %FALSE if not.
82
 *
83
 * Since: 2.28
84
 */
85
gboolean
86
g_pollable_input_stream_can_poll (GPollableInputStream *stream)
87
0
{
88
0
  g_return_val_if_fail (G_IS_POLLABLE_INPUT_STREAM (stream), FALSE);
89
90
0
  return G_POLLABLE_INPUT_STREAM_GET_INTERFACE (stream)->can_poll (stream);
91
0
}
92
93
/**
94
 * g_pollable_input_stream_is_readable:
95
 * @stream: a #GPollableInputStream.
96
 *
97
 * Checks if @stream can be read.
98
 *
99
 * Note that some stream types may not be able to implement this 100%
100
 * reliably, and it is possible that a call to g_input_stream_read()
101
 * after this returns %TRUE would still block. To guarantee
102
 * non-blocking behavior, you should always use
103
 * g_pollable_input_stream_read_nonblocking(), which will return a
104
 * %G_IO_ERROR_WOULD_BLOCK error rather than blocking.
105
 *
106
 * The behaviour of this method is undefined if
107
 * g_pollable_input_stream_can_poll() returns %FALSE for @stream.
108
 *
109
 * Returns: %TRUE if @stream is readable, %FALSE if not. If an error
110
 *   has occurred on @stream, this will result in
111
 *   g_pollable_input_stream_is_readable() returning %TRUE, and the
112
 *   next attempt to read will return the error.
113
 *
114
 * Since: 2.28
115
 */
116
gboolean
117
g_pollable_input_stream_is_readable (GPollableInputStream *stream)
118
0
{
119
0
  g_return_val_if_fail (G_IS_POLLABLE_INPUT_STREAM (stream), FALSE);
120
121
0
  return G_POLLABLE_INPUT_STREAM_GET_INTERFACE (stream)->is_readable (stream);
122
0
}
123
124
/**
125
 * g_pollable_input_stream_create_source:
126
 * @stream: a #GPollableInputStream.
127
 * @cancellable: (nullable): a #GCancellable, or %NULL
128
 *
129
 * Creates a #GSource that triggers when @stream can be read, or
130
 * @cancellable is triggered or an error occurs. The callback on the
131
 * source is of the #GPollableSourceFunc type.
132
 *
133
 * As with g_pollable_input_stream_is_readable(), it is possible that
134
 * the stream may not actually be readable even after the source
135
 * triggers, so you should use g_pollable_input_stream_read_nonblocking()
136
 * rather than g_input_stream_read() from the callback.
137
 *
138
 * The behaviour of this method is undefined if
139
 * g_pollable_input_stream_can_poll() returns %FALSE for @stream.
140
 *
141
 * Returns: (transfer full): a new #GSource
142
 *
143
 * Since: 2.28
144
 */
145
GSource *
146
g_pollable_input_stream_create_source (GPollableInputStream *stream,
147
               GCancellable         *cancellable)
148
0
{
149
0
  g_return_val_if_fail (G_IS_POLLABLE_INPUT_STREAM (stream), NULL);
150
151
0
  return G_POLLABLE_INPUT_STREAM_GET_INTERFACE (stream)->
152
0
    create_source (stream, cancellable);
153
0
}
154
155
static gssize
156
g_pollable_input_stream_default_read_nonblocking (GPollableInputStream  *stream,
157
              void                  *buffer,
158
              gsize                  count,
159
              GError               **error)
160
0
{
161
0
  if (!g_pollable_input_stream_is_readable (stream))
162
0
    {
163
0
      g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_WOULD_BLOCK,
164
0
                           g_strerror (EAGAIN));
165
0
      return -1;
166
0
    }
167
168
0
  return G_INPUT_STREAM_GET_CLASS (stream)->
169
0
    read_fn (G_INPUT_STREAM (stream), buffer, count, NULL, error);
170
0
}
171
172
/**
173
 * g_pollable_input_stream_read_nonblocking:
174
 * @stream: a #GPollableInputStream
175
 * @buffer: (array length=count) (element-type guint8) (out caller-allocates): a
176
 *     buffer to read data into (which should be at least @count bytes long).
177
 * @count: (in): the number of bytes you want to read
178
 * @cancellable: (nullable): a #GCancellable, or %NULL
179
 * @error: #GError for error reporting, or %NULL to ignore.
180
 *
181
 * Attempts to read up to @count bytes from @stream into @buffer, as
182
 * with g_input_stream_read(). If @stream is not currently readable,
183
 * this will immediately return %G_IO_ERROR_WOULD_BLOCK, and you can
184
 * use g_pollable_input_stream_create_source() to create a #GSource
185
 * that will be triggered when @stream is readable.
186
 *
187
 * Note that since this method never blocks, you cannot actually
188
 * use @cancellable to cancel it. However, it will return an error
189
 * if @cancellable has already been cancelled when you call, which
190
 * may happen if you call this method after a source triggers due
191
 * to having been cancelled.
192
 *
193
 * The behaviour of this method is undefined if
194
 * g_pollable_input_stream_can_poll() returns %FALSE for @stream.
195
 *
196
 * Virtual: read_nonblocking
197
 * Returns: the number of bytes read, or -1 on error (including
198
 *   %G_IO_ERROR_WOULD_BLOCK).
199
 */
200
gssize
201
g_pollable_input_stream_read_nonblocking (GPollableInputStream  *stream,
202
            void                  *buffer,
203
            gsize                  count,
204
            GCancellable          *cancellable,
205
            GError               **error)
206
0
{
207
0
  gssize res;
208
209
0
  g_return_val_if_fail (G_IS_POLLABLE_INPUT_STREAM (stream), -1);
210
0
  g_return_val_if_fail (buffer != NULL, 0);
211
212
0
  if (g_cancellable_set_error_if_cancelled (cancellable, error))
213
0
    return -1;
214
215
0
  if (count == 0)
216
0
    return 0;
217
218
0
  if (((gssize) count) < 0)
219
0
    {
220
0
      g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
221
0
       _("Too large count value passed to %s"), G_STRFUNC);
222
0
      return -1;
223
0
    }
224
225
0
  if (cancellable)
226
0
    g_cancellable_push_current (cancellable);
227
228
0
  res = G_POLLABLE_INPUT_STREAM_GET_INTERFACE (stream)->
229
0
    read_nonblocking (stream, buffer, count, error);
230
231
0
  if (cancellable)
232
0
    g_cancellable_pop_current (cancellable);
233
234
0
  return res;
235
0
}