xref: /dflybsd-src/contrib/binutils-2.34/ld/ldbuildid.c (revision b52ef7118d1621abed722c5bbbd542210290ecef)
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