[v3,0/7] Fix some libm static issues
[platform/upstream/glibc.git] / scripts / versions.awk
1 # Combine version map fragments into version scripts for our shared objects.
2 # Copyright (C) 1998-2024 Free Software Foundation, Inc.
3
4 # This script expects the following variables to be defined:
5 # defsfile              name of Versions.def file
6 # buildroot             name of build directory with trailing slash
7 # move_if_change        move-if-change command
8
9 # Read definitions for the versions.
10 BEGIN {
11   lossage = 0;
12
13   nlibs=0;
14   while (getline < defsfile) {
15     if (/^[a-zA-Z0-9_.]+ \{/) {
16       libs[$1] = 1;
17       curlib = $1;
18       while (getline < defsfile && ! /^}/) {
19         if ($2 == "=") {
20           renamed[curlib "::" $1] = $3;
21         }
22         else
23           versions[curlib "::" $1] = 1;
24       }
25     }
26   }
27   close(defsfile);
28
29   tmpfile = buildroot "Versions.tmp";
30   # POSIX sort needed.
31   sort = "sort -t. -k 1,1 -k 2n,2n -k 3 > " tmpfile;
32 }
33
34 # GNU awk does not implement the ord and chr functions.
35 # <https://www.gnu.org/software/gawk/manual/html_node/Ordinal-Functions.html>
36 # says that they are "written very nicely", using code similar to what
37 # is included here.
38 function chr(c) {
39     return sprintf("%c", c)
40 }
41
42 BEGIN {
43     for (c = 1; c < 127; c++) {
44         ord_table[chr(c)] = c;
45     }
46 }
47
48 function ord(c) {
49     if (ord_table[c]) {
50         return ord_table[c];
51     } else {
52         printf("Invalid character reference: '%c'\n", c) > "/dev/stderr";
53         ++lossage;
54     }
55 }
56
57 # Remove comment lines.
58 /^ *#/ {
59   next;
60 }
61
62 # This matches the beginning of the version information for a new library.
63 /^[a-zA-Z0-9_.]+/ {
64   actlib = $1;
65   if (!libs[$1]) {
66     printf("no versions defined for %s\n", $1) > "/dev/stderr";
67     ++lossage;
68   }
69   next;
70 }
71
72 # This matches the beginning of a new version for the current library.
73 /^  [A-Za-z_]/ {
74   if (renamed[actlib "::" $1])
75     actver = renamed[actlib "::" $1];
76   else if (!versions[actlib "::" $1] && $1 != "GLIBC_PRIVATE") {
77     printf("version %s not defined for %s\n", $1, actlib) > "/dev/stderr";
78     ++lossage;
79   }
80   else
81     actver = $1;
82   next;
83 }
84
85 # This matches lines with names to be added to the current version in the
86 # current library.  This is the only place where we print something to
87 # the intermediate file.
88 /^   / {
89   sortver=actver
90   # Ensure GLIBC_ versions come always first
91   sub(/^GLIBC_/," GLIBC_",sortver)
92   printf("%s %s %s\n", actlib, sortver, $0) | sort;
93 }
94
95 # Some targets do not set the ABI baseline for libdl.  As a result,
96 # symbols originally in libdl need to be moved under historic symbol
97 # versions, without altering the baseline version for libc itself.
98 /^ *!libc_pre_versions/ {
99     libc_pre_versions_active = 1;
100 }
101
102 function libc_pre_versions() {
103     # No local: * here, so that we do not have to update this script
104     # if symbols are moved into libc.  The abilist files and the other
105     # targets (with a real GLIBC_2.0 baseline) provide testing
106     # coverage.
107     printf("\
108 GLIBC_2.0 {\n\
109 };\n\
110 GLIBC_2.1 {\n\
111 } GLIBC_2.0;\n\
112 ") > outfile;
113     return "GLIBC_2.1";
114 }
115
116 function closeversion(name, oldname) {
117   printf("  local:\n    *;\n") > outfile;
118   # This version inherits from the last one only if they
119   # have the same nonnumeric prefix, i.e. GLIBC_x.y and GLIBC_x.z
120   # or FOO_x and FOO_y but not GLIBC_x and FOO_y.
121   pfx = oldname;
122   sub(/[0-9.]+/,".+",pfx);
123   if (oldname == "" || name !~ pfx) print "};" > outfile;
124   else printf("} %s;\n", oldname) > outfile;
125 }
126
127 function close_and_move(name, real_name) {
128   close(name);
129   system(move_if_change " " name " " real_name " >&2");
130 }
131
132 # ELF hash, for use with symbol versions.
133 function elf_hash(s, i, acc) {
134   acc = 0;
135   for (i = 1; i <= length(s); ++i) {
136       acc = and(lshift(acc, 4) + ord(substr(s, i, 1)), 0xffffffff);
137       top = and(acc, 0xf0000000);
138       acc = and(xor(acc, rshift(top, 24)), compl(top));
139   }
140   return acc;
141 }
142
143 # Now print the accumulated information.
144 END {
145   close(sort);
146
147   if (lossage) {
148     system("rm -f " tmpfile);
149     exit 1;
150   }
151
152   oldlib = "";
153   oldver = "";
154   real_first_ver_header = buildroot "first-versions.h"
155   first_ver_header = real_first_ver_header "T"
156   printf("#ifndef _FIRST_VERSIONS_H\n") > first_ver_header;
157   printf("#define _FIRST_VERSIONS_H\n") > first_ver_header;
158   real_ldbl_compat_header = buildroot "ldbl-compat-choose.h"
159   ldbl_compat_header = real_ldbl_compat_header "T"
160   printf("#ifndef _LDBL_COMPAT_CHOOSE_H\n") > ldbl_compat_header;
161   printf("#define _LDBL_COMPAT_CHOOSE_H\n") > ldbl_compat_header;
162   printf("#ifndef LONG_DOUBLE_COMPAT\n") > ldbl_compat_header;
163   printf("# error LONG_DOUBLE_COMPAT not defined\n") > ldbl_compat_header;
164   printf("#endif\n") > ldbl_compat_header;
165   printf("version-maps =");
166   while (getline < tmpfile) {
167     if ($1 != oldlib) {
168       if (oldlib != "") {
169         closeversion(oldver, veryoldver);
170         oldver = "";
171         close_and_move(outfile, real_outfile);
172       }
173       oldlib = $1;
174       real_outfile = buildroot oldlib ".map";
175       outfile = real_outfile "T";
176       if ($1 == "libc" && libc_pre_versions_active) {
177           veryoldver = libc_pre_versions();
178       } else {
179           veryoldver = "";
180       }
181       printf(" %s.map", oldlib);
182     }
183     if ($2 != oldver) {
184       if (oldver != "") {
185         closeversion(oldver, veryoldver);
186         veryoldver = oldver;
187       }
188       oldver = $2;
189       # Skip the placeholder symbol used only for empty version map.
190       if ($3 == "__placeholder_only_for_empty_version_map;") {
191         printf("%s {\n", $2) > outfile;
192         continue;
193       }
194       printf("%s {\n  global:\n", $2) > outfile;
195     }
196     printf("   ") > outfile;
197     for (n = 3; n <= NF; ++n) {
198       printf(" %s", $n) > outfile;
199       sym = $n;
200       sub(";", "", sym);
201       first_ver_macro = "FIRST_VERSION_" oldlib "_" sym;
202       if (!(first_ver_macro in first_ver_seen) \
203           && oldver ~ "^GLIBC_[0-9]" \
204           && sym ~ "^[A-Za-z0-9_]*$") {
205         ver_val = oldver;
206         printf("#define %s_STRING \"%s\"\n", first_ver_macro, ver_val) > first_ver_header;
207         printf("#define %s_HASH 0x%x\n", first_ver_macro, elf_hash(ver_val)) > first_ver_header;
208         gsub("\\.", "_", ver_val);
209         printf("#define %s %s\n", first_ver_macro, ver_val) > first_ver_header;
210         first_ver_seen[first_ver_macro] = 1;
211         if (oldlib == "libc" || oldlib == "libm") {
212           printf("#if LONG_DOUBLE_COMPAT (%s, %s)\n",
213                  oldlib, ver_val) > ldbl_compat_header;
214           printf("# define LONG_DOUBLE_COMPAT_CHOOSE_%s_%s(a, b) a\n",
215                  oldlib, sym) > ldbl_compat_header;
216           printf("#else\n") > ldbl_compat_header;
217           printf("# define LONG_DOUBLE_COMPAT_CHOOSE_%s_%s(a, b) b\n",
218                  oldlib, sym) > ldbl_compat_header;
219           printf("#endif\n") > ldbl_compat_header;
220         }
221       }
222     }
223     printf("\n") > outfile;
224   }
225   printf("\n");
226   printf("#endif /* first-versions.h */\n") > first_ver_header;
227   printf("#endif /* ldbl-compat-choose.h */\n") > ldbl_compat_header;
228   closeversion(oldver, veryoldver);
229   close_and_move(outfile, real_outfile);
230   close_and_move(first_ver_header, real_first_ver_header);
231   close_and_move(ldbl_compat_header, real_ldbl_compat_header);
232   #system("rm -f " tmpfile);
233 }