1 /* Copyright (C) 1991-2022 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
3
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
8
9 The GNU C Library 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 GNU
12 Lesser General Public License for more details.
13
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, see
16 <https://www.gnu.org/licenses/>. */
17
18 #include <errno.h>
19 #include <limits.h>
20 #include <printf.h>
21 #include <stddef.h>
22 #include <stdlib.h>
23 #include <libc-lock.h>
24
25
26 /* Array of functions indexed by format character. */
27 libc_freeres_ptr (printf_arginfo_size_function **__printf_arginfo_table)
28 attribute_hidden;
29 printf_function **__printf_function_table attribute_hidden;
30
__libc_lock_define_initialized(static,lock)31 __libc_lock_define_initialized (static, lock)
32
33 /* Register FUNC to be called to format SPEC specifiers. */
34 int
35 __register_printf_specifier (int spec, printf_function converter,
36 printf_arginfo_size_function arginfo)
37 {
38 if (spec < 0 || spec > (int) UCHAR_MAX)
39 {
40 __set_errno (EINVAL);
41 return -1;
42 }
43
44 int result = 0;
45 __libc_lock_lock (lock);
46
47 if (__printf_function_table == NULL)
48 {
49 __printf_arginfo_table = (printf_arginfo_size_function **)
50 calloc (UCHAR_MAX + 1, sizeof (void *) * 2);
51 if (__printf_arginfo_table == NULL)
52 {
53 result = -1;
54 goto out;
55 }
56
57 __printf_function_table = (printf_function **)
58 (__printf_arginfo_table + UCHAR_MAX + 1);
59 }
60
61 __printf_function_table[spec] = converter;
62 __printf_arginfo_table[spec] = arginfo;
63
64 out:
65 __libc_lock_unlock (lock);
66
67 return result;
68 }
69 libc_hidden_def (__register_printf_specifier)
weak_alias(__register_printf_specifier,register_printf_specifier)70 weak_alias (__register_printf_specifier, register_printf_specifier)
71
72
73 /* Register FUNC to be called to format SPEC specifiers. */
74 int
75 __register_printf_function (int spec, printf_function converter,
76 printf_arginfo_function arginfo)
77 {
78 return __register_printf_specifier (spec, converter,
79 (printf_arginfo_size_function*) arginfo);
80 }
81 weak_alias (__register_printf_function, register_printf_function)
82