Mercurial > audlegacy-plugins
annotate src/stdio/stdio.c @ 2971:3134a0987162
- changed include path from audacious to audlegacy.
- changed include path for libSAD.h.
author | Yoshiki Yazawa <yaz@honeyplanet.jp> |
---|---|
date | Wed, 26 Nov 2008 02:55:04 +0900 |
parents | 30d30cfe6b47 |
children |
rev | line source |
---|---|
306 | 1 /* Audacious |
2 * Copyright (c) 2006 William Pitcock | |
3 * | |
4 * This program is free software; you can redistribute it and/or modify | |
5 * it under the terms of the GNU General Public License as published by | |
6 * the Free Software Foundation; either version 2 of the License, or | |
7 * (at your option) any later version. | |
8 * | |
9 * This program is distributed in the hope that it will be useful, | |
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
12 * GNU General Public License for more details. | |
13 * | |
14 * You should have received a copy of the GNU General Public License | |
15 * along with this program; if not, write to the Free Software | |
16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | |
17 */ | |
18 | |
1957 | 19 #include "config.h" |
2971
3134a0987162
- changed include path from audacious to audlegacy.
Yoshiki Yazawa <yaz@honeyplanet.jp>
parents:
2940
diff
changeset
|
20 #include <audlegacy/plugin.h> |
306 | 21 #include <stdio.h> |
22 | |
23 #include <unistd.h> | |
24 #include <sys/stat.h> | |
25 #include <sys/types.h> | |
26 | |
500
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
27 #include <string.h> |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
28 |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
29 static gchar * |
1978 | 30 aud_vfs_stdio_urldecode_path(const gchar * encoded_path) |
500
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
31 { |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
32 const gchar *cur, *ext; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
33 gchar *path, *tmp; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
34 gint realchar; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
35 |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
36 if (!encoded_path) |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
37 return NULL; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
38 |
2050
2ffc6a69fcd1
string API calls -> vtable
William Pitcock <nenolod@atheme.org>
parents:
1980
diff
changeset
|
39 if (!aud_str_has_prefix_nocase(encoded_path, "file:")) |
500
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
40 return NULL; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
41 |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
42 cur = encoded_path + 5; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
43 |
2050
2ffc6a69fcd1
string API calls -> vtable
William Pitcock <nenolod@atheme.org>
parents:
1980
diff
changeset
|
44 if (aud_str_has_prefix_nocase(cur, "//localhost")) |
500
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
45 cur += 11; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
46 |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
47 if (*cur == '/') |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
48 while (cur[1] == '/') |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
49 cur++; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
50 |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
51 tmp = g_malloc0(strlen(cur) + 1); |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
52 |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
53 while ((ext = strchr(cur, '%')) != NULL) { |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
54 strncat(tmp, cur, ext - cur); |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
55 ext++; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
56 cur = ext + 2; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
57 if (!sscanf(ext, "%2x", &realchar)) { |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
58 /* Assume it is a literal '%'. Several file |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
59 * managers send unencoded file: urls on drag |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
60 * and drop. */ |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
61 realchar = '%'; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
62 cur -= 2; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
63 } |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
64 tmp[strlen(tmp)] = realchar; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
65 } |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
66 |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
67 path = g_strconcat(tmp, cur, NULL); |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
68 g_free(tmp); |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
69 return path; |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
70 } |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
71 |
306 | 72 VFSFile * |
1978 | 73 stdio_aud_vfs_fopen_impl(const gchar * path, |
306 | 74 const gchar * mode) |
75 { | |
76 VFSFile *file; | |
500
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
77 gchar *decpath; |
306 | 78 |
79 if (!path || !mode) | |
2940
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
80 return NULL; |
306 | 81 |
1978 | 82 decpath = aud_vfs_stdio_urldecode_path(path); |
500
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
83 |
306 | 84 file = g_new(VFSFile, 1); |
85 | |
500
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
86 file->handle = fopen(decpath != NULL ? decpath : path, mode); |
1d81ea250dce
[svn] - URIs are now passed naked instead of mangled by the VFS subsystem.
nenolod
parents:
326
diff
changeset
|
87 |
2940
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
88 g_free(decpath); |
306 | 89 |
90 if (file->handle == NULL) { | |
91 g_free(file); | |
92 file = NULL; | |
93 } | |
94 | |
95 return file; | |
96 } | |
97 | |
98 gint | |
1978 | 99 stdio_aud_vfs_fclose_impl(VFSFile * file) |
306 | 100 { |
101 gint ret = 0; | |
102 | |
103 if (file == NULL) | |
104 return -1; | |
105 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
106 if (file->handle) |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
107 { |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
108 FILE *handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
109 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
110 if (fclose(handle) != 0) |
306 | 111 ret = -1; |
1228
01efa0e5b554
file->handle should be cleared on close, otherwise it can lead to double free.
Yoshiki Yazawa <yaz@cc.rim.or.jp>
parents:
1210
diff
changeset
|
112 file->handle = NULL; |
306 | 113 } |
114 | |
115 return ret; | |
116 } | |
117 | |
118 size_t | |
1978 | 119 stdio_aud_vfs_fread_impl(gpointer ptr, |
306 | 120 size_t size, |
121 size_t nmemb, | |
122 VFSFile * file) | |
123 { | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
124 FILE *handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
125 |
306 | 126 if (file == NULL) |
127 return 0; | |
128 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
129 handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
130 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
131 return fread(ptr, size, nmemb, handle); |
306 | 132 } |
133 | |
134 size_t | |
1978 | 135 stdio_aud_vfs_fwrite_impl(gconstpointer ptr, |
306 | 136 size_t size, |
137 size_t nmemb, | |
138 VFSFile * file) | |
139 { | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
140 FILE *handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
141 |
306 | 142 if (file == NULL) |
143 return 0; | |
144 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
145 handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
146 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
147 return fwrite(ptr, size, nmemb, handle); |
306 | 148 } |
149 | |
150 gint | |
1978 | 151 stdio_aud_vfs_getc_impl(VFSFile *stream) |
306 | 152 { |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
153 FILE *handle = (FILE *) stream->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
154 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
155 return getc( handle ); |
306 | 156 } |
157 | |
158 gint | |
2940
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
159 stdio_aud_vfs_ungetc_impl(gint c, VFSFile * file) |
306 | 160 { |
2940
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
161 FILE *handle; |
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
162 |
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
163 if (file == NULL) |
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
164 return -1; |
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
165 |
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
166 handle = (FILE *) file->handle; |
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
167 |
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
168 return ungetc(c, handle); |
306 | 169 } |
170 | |
171 gint | |
1978 | 172 stdio_aud_vfs_fseek_impl(VFSFile * file, |
306 | 173 glong offset, |
174 gint whence) | |
175 { | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
176 FILE *handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
177 |
306 | 178 if (file == NULL) |
179 return 0; | |
180 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
181 handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
182 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
183 return fseek(handle, offset, whence); |
306 | 184 } |
185 | |
186 void | |
1978 | 187 stdio_aud_vfs_rewind_impl(VFSFile * file) |
306 | 188 { |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
189 FILE *handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
190 |
306 | 191 if (file == NULL) |
192 return; | |
193 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
194 handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
195 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
196 rewind(handle); |
306 | 197 } |
198 | |
199 glong | |
1978 | 200 stdio_aud_vfs_ftell_impl(VFSFile * file) |
306 | 201 { |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
202 FILE *handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
203 |
306 | 204 if (file == NULL) |
205 return 0; | |
206 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
207 handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
208 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
209 return ftell(handle); |
306 | 210 } |
211 | |
212 gboolean | |
1978 | 213 stdio_aud_vfs_feof_impl(VFSFile * file) |
306 | 214 { |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
215 FILE *handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
216 |
306 | 217 if (file == NULL) |
218 return FALSE; | |
219 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
220 handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
221 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
222 return (gboolean) feof(handle); |
306 | 223 } |
224 | |
225 gint | |
1978 | 226 stdio_aud_vfs_truncate_impl(VFSFile * file, glong size) |
306 | 227 { |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
228 FILE *handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
229 |
306 | 230 if (file == NULL) |
231 return -1; | |
232 | |
326
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
233 handle = (FILE *) file->handle; |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
234 |
90a843d02970
[svn] - some OSes use macros for some f*() functions, therefore make sure we
nenolod
parents:
306
diff
changeset
|
235 return ftruncate(fileno(handle), size); |
306 | 236 } |
237 | |
965 | 238 off_t |
1978 | 239 stdio_aud_vfs_fsize_impl(VFSFile * file) |
965 | 240 { |
241 FILE *handle; | |
242 struct stat s; | |
243 | |
244 if (file == NULL) | |
245 return -1; | |
246 | |
247 handle = (FILE *) file->handle; | |
248 | |
2940
30d30cfe6b47
Handle file == NULL in ungetc().
Matti Hamalainen <ccr@tnsp.org>
parents:
2576
diff
changeset
|
249 if (fstat(fileno(handle), &s) == -1) |
965 | 250 return -1; |
251 | |
252 return s.st_size; | |
253 } | |
254 | |
306 | 255 VFSConstructor file_const = { |
1957 | 256 .uri_id = "file://", |
1980 | 257 .vfs_fopen_impl = stdio_aud_vfs_fopen_impl, |
258 .vfs_fclose_impl = stdio_aud_vfs_fclose_impl, | |
259 .vfs_fread_impl = stdio_aud_vfs_fread_impl, | |
260 .vfs_fwrite_impl = stdio_aud_vfs_fwrite_impl, | |
261 .vfs_getc_impl = stdio_aud_vfs_getc_impl, | |
262 .vfs_ungetc_impl = stdio_aud_vfs_ungetc_impl, | |
263 .vfs_fseek_impl = stdio_aud_vfs_fseek_impl, | |
264 .vfs_rewind_impl = stdio_aud_vfs_rewind_impl, | |
265 .vfs_ftell_impl = stdio_aud_vfs_ftell_impl, | |
266 .vfs_feof_impl = stdio_aud_vfs_feof_impl, | |
267 .vfs_truncate_impl = stdio_aud_vfs_truncate_impl, | |
268 .vfs_fsize_impl = stdio_aud_vfs_fsize_impl | |
306 | 269 }; |
270 | |
271 static void init(void) | |
272 { | |
1978 | 273 aud_vfs_register_transport(&file_const); |
306 | 274 } |
275 | |
276 static void cleanup(void) | |
277 { | |
278 #if 0 | |
1978 | 279 aud_vfs_unregister_transport(&file_const); |
306 | 280 #endif |
281 } | |
282 | |
1395
761e17b23e0c
added Discovery plugin type
Cristi Magherusan <majeru@atheme-project.org>
parents:
1228
diff
changeset
|
283 DECLARE_PLUGIN(stdio, init, cleanup, NULL, NULL, NULL, NULL, NULL, NULL); |