1*fae548d3Szrj /* ldbuildid.c - Build Id support routines
2*fae548d3Szrj Copyright (C) 2013-2020 Free Software Foundation, Inc.
3*fae548d3Szrj
4*fae548d3Szrj This file is part of the GNU Binutils.
5*fae548d3Szrj
6*fae548d3Szrj This program is free software; you can redistribute it and/or modify
7*fae548d3Szrj it under the terms of the GNU General Public License as published by
8*fae548d3Szrj the Free Software Foundation; either version 3 of the License, or
9*fae548d3Szrj (at your option) any later version.
10*fae548d3Szrj
11*fae548d3Szrj This program is distributed in the hope that it will be useful,
12*fae548d3Szrj but WITHOUT ANY WARRANTY; without even the implied warranty of
13*fae548d3Szrj MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14*fae548d3Szrj GNU General Public License for more details.
15*fae548d3Szrj
16*fae548d3Szrj You should have received a copy of the GNU General Public License
17*fae548d3Szrj along with this program; if not, write to the Free Software
18*fae548d3Szrj Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
19*fae548d3Szrj MA 02110-1301, USA. */
20*fae548d3Szrj
21*fae548d3Szrj #include "sysdep.h"
22*fae548d3Szrj #include "bfd.h"
23*fae548d3Szrj #include "safe-ctype.h"
24*fae548d3Szrj #include "md5.h"
25*fae548d3Szrj #include "sha1.h"
26*fae548d3Szrj #include "ldbuildid.h"
27*fae548d3Szrj #ifdef __MINGW32__
28*fae548d3Szrj #include <windows.h>
29*fae548d3Szrj #include <rpcdce.h>
30*fae548d3Szrj #endif
31*fae548d3Szrj
32*fae548d3Szrj #define streq(a,b) strcmp ((a), (b)) == 0
33*fae548d3Szrj #define strneq(a,b,n) strncmp ((a), (b), (n)) == 0
34*fae548d3Szrj
35*fae548d3Szrj bfd_boolean
validate_build_id_style(const char * style)36*fae548d3Szrj validate_build_id_style (const char *style)
37*fae548d3Szrj {
38*fae548d3Szrj if ((streq (style, "md5")) || (streq (style, "sha1"))
39*fae548d3Szrj || (streq (style, "uuid")) || (strneq (style, "0x", 2)))
40*fae548d3Szrj return TRUE;
41*fae548d3Szrj
42*fae548d3Szrj return FALSE;
43*fae548d3Szrj }
44*fae548d3Szrj
45*fae548d3Szrj bfd_size_type
compute_build_id_size(const char * style)46*fae548d3Szrj compute_build_id_size (const char *style)
47*fae548d3Szrj {
48*fae548d3Szrj if (streq (style, "md5") || streq (style, "uuid"))
49*fae548d3Szrj return 128 / 8;
50*fae548d3Szrj
51*fae548d3Szrj if (streq (style, "sha1"))
52*fae548d3Szrj return 160 / 8;
53*fae548d3Szrj
54*fae548d3Szrj if (strneq (style, "0x", 2))
55*fae548d3Szrj {
56*fae548d3Szrj bfd_size_type size = 0;
57*fae548d3Szrj /* ID is in string form (hex). Count the bytes. */
58*fae548d3Szrj const char *id = style + 2;
59*fae548d3Szrj
60*fae548d3Szrj do
61*fae548d3Szrj {
62*fae548d3Szrj if (ISXDIGIT (id[0]) && ISXDIGIT (id[1]))
63*fae548d3Szrj {
64*fae548d3Szrj ++size;
65*fae548d3Szrj id += 2;
66*fae548d3Szrj }
67*fae548d3Szrj else if (*id == '-' || *id == ':')
68*fae548d3Szrj ++id;
69*fae548d3Szrj else
70*fae548d3Szrj {
71*fae548d3Szrj size = 0;
72*fae548d3Szrj break;
73*fae548d3Szrj }
74*fae548d3Szrj } while (*id != '\0');
75*fae548d3Szrj return size;
76*fae548d3Szrj }
77*fae548d3Szrj
78*fae548d3Szrj return 0;
79*fae548d3Szrj }
80*fae548d3Szrj
81*fae548d3Szrj static unsigned char
read_hex(const char xdigit)82*fae548d3Szrj read_hex (const char xdigit)
83*fae548d3Szrj {
84*fae548d3Szrj if (ISDIGIT (xdigit))
85*fae548d3Szrj return xdigit - '0';
86*fae548d3Szrj
87*fae548d3Szrj if (ISUPPER (xdigit))
88*fae548d3Szrj return xdigit - 'A' + 0xa;
89*fae548d3Szrj
90*fae548d3Szrj if (ISLOWER (xdigit))
91*fae548d3Szrj return xdigit - 'a' + 0xa;
92*fae548d3Szrj
93*fae548d3Szrj abort ();
94*fae548d3Szrj return 0;
95*fae548d3Szrj }
96*fae548d3Szrj
97*fae548d3Szrj bfd_boolean
generate_build_id(bfd * abfd,const char * style,checksum_fn checksum_contents,unsigned char * id_bits,int size ATTRIBUTE_UNUSED)98*fae548d3Szrj generate_build_id (bfd *abfd,
99*fae548d3Szrj const char *style,
100*fae548d3Szrj checksum_fn checksum_contents,
101*fae548d3Szrj unsigned char *id_bits,
102*fae548d3Szrj int size ATTRIBUTE_UNUSED)
103*fae548d3Szrj {
104*fae548d3Szrj if (streq (style, "md5"))
105*fae548d3Szrj {
106*fae548d3Szrj struct md5_ctx ctx;
107*fae548d3Szrj
108*fae548d3Szrj md5_init_ctx (&ctx);
109*fae548d3Szrj if (!(*checksum_contents) (abfd, (sum_fn) &md5_process_bytes, &ctx))
110*fae548d3Szrj return FALSE;
111*fae548d3Szrj md5_finish_ctx (&ctx, id_bits);
112*fae548d3Szrj }
113*fae548d3Szrj else if (streq (style, "sha1"))
114*fae548d3Szrj {
115*fae548d3Szrj struct sha1_ctx ctx;
116*fae548d3Szrj
117*fae548d3Szrj sha1_init_ctx (&ctx);
118*fae548d3Szrj if (!(*checksum_contents) (abfd, (sum_fn) &sha1_process_bytes, &ctx))
119*fae548d3Szrj return FALSE;
120*fae548d3Szrj sha1_finish_ctx (&ctx, id_bits);
121*fae548d3Szrj }
122*fae548d3Szrj else if (streq (style, "uuid"))
123*fae548d3Szrj {
124*fae548d3Szrj #ifndef __MINGW32__
125*fae548d3Szrj int n;
126*fae548d3Szrj int fd = open ("/dev/urandom", O_RDONLY);
127*fae548d3Szrj
128*fae548d3Szrj if (fd < 0)
129*fae548d3Szrj return FALSE;
130*fae548d3Szrj n = read (fd, id_bits, size);
131*fae548d3Szrj close (fd);
132*fae548d3Szrj if (n < size)
133*fae548d3Szrj return FALSE;
134*fae548d3Szrj #else /* __MINGW32__ */
135*fae548d3Szrj typedef RPC_STATUS (RPC_ENTRY * UuidCreateFn) (UUID *);
136*fae548d3Szrj UUID uuid;
137*fae548d3Szrj UuidCreateFn uuid_create = 0;
138*fae548d3Szrj HMODULE rpc_library = LoadLibrary ("rpcrt4.dll");
139*fae548d3Szrj
140*fae548d3Szrj if (!rpc_library)
141*fae548d3Szrj return FALSE;
142*fae548d3Szrj uuid_create = (UuidCreateFn) (void (WINAPI *)(void)) GetProcAddress (rpc_library, "UuidCreate");
143*fae548d3Szrj if (!uuid_create)
144*fae548d3Szrj {
145*fae548d3Szrj FreeLibrary (rpc_library);
146*fae548d3Szrj return FALSE;
147*fae548d3Szrj }
148*fae548d3Szrj
149*fae548d3Szrj if (uuid_create (&uuid) != RPC_S_OK)
150*fae548d3Szrj {
151*fae548d3Szrj FreeLibrary (rpc_library);
152*fae548d3Szrj return FALSE;
153*fae548d3Szrj }
154*fae548d3Szrj FreeLibrary (rpc_library);
155*fae548d3Szrj memcpy (id_bits, &uuid,
156*fae548d3Szrj (size_t) size < sizeof (UUID) ? (size_t) size : sizeof (UUID));
157*fae548d3Szrj #endif /* __MINGW32__ */
158*fae548d3Szrj }
159*fae548d3Szrj else if (strneq (style, "0x", 2))
160*fae548d3Szrj {
161*fae548d3Szrj /* ID is in string form (hex). Convert to bits. */
162*fae548d3Szrj const char *id = style + 2;
163*fae548d3Szrj size_t n = 0;
164*fae548d3Szrj
165*fae548d3Szrj do
166*fae548d3Szrj {
167*fae548d3Szrj if (ISXDIGIT (id[0]) && ISXDIGIT (id[1]))
168*fae548d3Szrj {
169*fae548d3Szrj id_bits[n] = read_hex (*id++) << 4;
170*fae548d3Szrj id_bits[n++] |= read_hex (*id++);
171*fae548d3Szrj }
172*fae548d3Szrj else if (*id == '-' || *id == ':')
173*fae548d3Szrj ++id;
174*fae548d3Szrj else
175*fae548d3Szrj abort (); /* Should have been validated earlier. */
176*fae548d3Szrj }
177*fae548d3Szrj while (*id != '\0');
178*fae548d3Szrj }
179*fae548d3Szrj else
180*fae548d3Szrj abort (); /* Should have been validated earlier. */
181*fae548d3Szrj
182*fae548d3Szrj return TRUE;
183*fae548d3Szrj }
184