msvcrt: Added _fscanf_l implementation.
[wine] / dlls / msvcrt / scanf.c
1 /*
2  * general implementation of scanf used by scanf, sscanf, fscanf,
3  * _cscanf, wscanf, swscanf and fwscanf
4  *
5  * Copyright 1996,1998 Marcus Meissner
6  * Copyright 1996 Jukka Iivonen
7  * Copyright 1997,2000 Uwe Bonnes
8  * Copyright 2000 Jon Griffiths
9  * Copyright 2002 Daniel Gudbjartsson
10  *
11  * This library is free software; you can redistribute it and/or
12  * modify it under the terms of the GNU Lesser General Public
13  * License as published by the Free Software Foundation; either
14  * version 2.1 of the License, or (at your option) any later version.
15  *
16  * This library is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
19  * Lesser General Public License for more details.
20  *
21  * You should have received a copy of the GNU Lesser General Public
22  * License along with this library; if not, write to the Free Software
23  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
24  */
25
26 #include <stdarg.h>
27
28 #include "windef.h"
29 #include "winbase.h"
30 #include "winternl.h"
31 #include "msvcrt.h"
32 #include "wine/debug.h"
33
34 WINE_DEFAULT_DEBUG_CHANNEL(msvcrt);
35
36 extern MSVCRT_FILE MSVCRT__iob[];
37
38 /* helper function for *scanf.  Returns the value of character c in the
39  * given base, or -1 if the given character is not a digit of the base.
40  */
41 static int char2digit(char c, int base) {
42     if ((c>='0') && (c<='9') && (c<='0'+base-1)) return (c-'0');
43     if (base<=10) return -1;
44     if ((c>='A') && (c<='Z') && (c<='A'+base-11)) return (c-'A'+10);
45     if ((c>='a') && (c<='z') && (c<='a'+base-11)) return (c-'a'+10);
46     return -1;
47 }
48
49 /* helper function for *wscanf.  Returns the value of character c in the
50  * given base, or -1 if the given character is not a digit of the base.
51  */
52 static int wchar2digit(MSVCRT_wchar_t c, int base) {
53     if ((c>='0') && (c<='9') && (c<='0'+base-1)) return (c-'0');
54     if (base<=10) return -1;
55     if ((c>='A') && (c<='Z') && (c<='A'+base-11)) return (c-'A'+10);
56     if ((c>='a') && (c<='z') && (c<='a'+base-11)) return (c-'a'+10);
57     return -1;
58 }
59
60 /* vfscanf_l */
61 #undef WIDE_SCANF
62 #undef CONSOLE
63 #undef STRING
64 #include "scanf.h"
65
66 /* vfwscanf_l */
67 #define WIDE_SCANF 1
68 #undef CONSOLE
69 #undef STRING
70 #include "scanf.h"
71
72 /* vsscanf_l */
73 #undef WIDE_SCANF
74 #undef CONSOLE
75 #define STRING 1
76 #include "scanf.h"
77
78 /* vswscanf_l */
79 #define WIDE_SCANF 1
80 #undef CONSOLE
81 #define STRING 1
82 #include "scanf.h"
83
84 /* vcscanf_l */
85 #undef WIDE_SCANF
86 #define CONSOLE 1
87 #undef STRING
88 #include "scanf.h"
89
90
91 /*********************************************************************
92  *              fscanf (MSVCRT.@)
93  */
94 int CDECL MSVCRT_fscanf(MSVCRT_FILE *file, const char *format, ...)
95 {
96     __ms_va_list valist;
97     int res;
98
99     __ms_va_start(valist, format);
100     res = MSVCRT_vfscanf_l(file, format, NULL, valist);
101     __ms_va_end(valist);
102     return res;
103 }
104
105 /*********************************************************************
106  *              _fscanf_l (MSVCRT.@)
107  */
108 int CDECL MSVCRT__fscanf_l(MSVCRT_FILE *file, const char *format,
109         MSVCRT__locale_t locale, ...)
110 {
111     __ms_va_list valist;
112     int res;
113
114     __ms_va_start(valist, locale);
115     res = MSVCRT_vfscanf_l(file, format, locale, valist);
116     __ms_va_end(valist);
117     return res;
118 }
119
120 /*********************************************************************
121  *              scanf (MSVCRT.@)
122  */
123 int CDECL MSVCRT_scanf(const char *format, ...)
124 {
125     __ms_va_list valist;
126     int res;
127
128     __ms_va_start(valist, format);
129     res = MSVCRT_vfscanf_l(MSVCRT_stdin, format, NULL, valist);
130     __ms_va_end(valist);
131     return res;
132 }
133
134 /*********************************************************************
135  *              fwscanf (MSVCRT.@)
136  */
137 int CDECL MSVCRT_fwscanf(MSVCRT_FILE *file, const MSVCRT_wchar_t *format, ...)
138 {
139     __ms_va_list valist;
140     int res;
141
142     __ms_va_start(valist, format);
143     res = MSVCRT_vfwscanf_l(file, format, NULL, valist);
144     __ms_va_end(valist);
145     return res;
146 }
147
148
149 /*********************************************************************
150  *              wscanf (MSVCRT.@)
151  */
152 int CDECL MSVCRT_wscanf(const MSVCRT_wchar_t *format, ...)
153 {
154     __ms_va_list valist;
155     int res;
156
157     __ms_va_start(valist, format);
158     res = MSVCRT_vfwscanf_l(MSVCRT_stdin, format, NULL, valist);
159     __ms_va_end(valist);
160     return res;
161 }
162
163
164 /*********************************************************************
165  *              sscanf (MSVCRT.@)
166  */
167 int CDECL MSVCRT_sscanf(const char *str, const char *format, ...)
168 {
169     __ms_va_list valist;
170     int res;
171
172     __ms_va_start(valist, format);
173     res = MSVCRT_vsscanf_l(str, format, NULL, valist);
174     __ms_va_end(valist);
175     return res;
176 }
177
178
179 /*********************************************************************
180  *              swscanf (MSVCRT.@)
181  */
182 int CDECL MSVCRT_swscanf(const MSVCRT_wchar_t *str, const MSVCRT_wchar_t *format, ...)
183 {
184     __ms_va_list valist;
185     int res;
186
187     __ms_va_start(valist, format);
188     res = MSVCRT_vswscanf_l(str, format, NULL, valist);
189     __ms_va_end(valist);
190     return res;
191 }
192
193
194 /*********************************************************************
195  *              _cscanf (MSVCRT.@)
196  */
197 int CDECL _cscanf(const char *format, ...)
198 {
199     __ms_va_list valist;
200     int res;
201
202     __ms_va_start(valist, format);
203     res = MSVCRT_vcscanf_l(format, NULL, valist);
204     __ms_va_end(valist);
205     return res;
206 }