blob: 01f17fe5c906e6f8dd225b7c07bfb725a54a48a8 [file] [log] [blame]
Elijah Newrenb73ecb42023-02-24 00:09:26 +00001#include "git-compat-util.h"
Elijah Newrend1cbe1e2023-04-22 20:17:20 +00002#include "hash.h"
Elijah Newrenb73ecb42023-02-24 00:09:26 +00003#include "hex.h"
Linus Torvaldsa5031212010-01-21 15:25:19 -08004
5const signed char hexval_table[256] = {
6 -1, -1, -1, -1, -1, -1, -1, -1, /* 00-07 */
7 -1, -1, -1, -1, -1, -1, -1, -1, /* 08-0f */
8 -1, -1, -1, -1, -1, -1, -1, -1, /* 10-17 */
9 -1, -1, -1, -1, -1, -1, -1, -1, /* 18-1f */
10 -1, -1, -1, -1, -1, -1, -1, -1, /* 20-27 */
11 -1, -1, -1, -1, -1, -1, -1, -1, /* 28-2f */
12 0, 1, 2, 3, 4, 5, 6, 7, /* 30-37 */
13 8, 9, -1, -1, -1, -1, -1, -1, /* 38-3f */
14 -1, 10, 11, 12, 13, 14, 15, -1, /* 40-47 */
15 -1, -1, -1, -1, -1, -1, -1, -1, /* 48-4f */
16 -1, -1, -1, -1, -1, -1, -1, -1, /* 50-57 */
17 -1, -1, -1, -1, -1, -1, -1, -1, /* 58-5f */
18 -1, 10, 11, 12, 13, 14, 15, -1, /* 60-67 */
19 -1, -1, -1, -1, -1, -1, -1, -1, /* 68-67 */
20 -1, -1, -1, -1, -1, -1, -1, -1, /* 70-77 */
21 -1, -1, -1, -1, -1, -1, -1, -1, /* 78-7f */
22 -1, -1, -1, -1, -1, -1, -1, -1, /* 80-87 */
23 -1, -1, -1, -1, -1, -1, -1, -1, /* 88-8f */
24 -1, -1, -1, -1, -1, -1, -1, -1, /* 90-97 */
25 -1, -1, -1, -1, -1, -1, -1, -1, /* 98-9f */
26 -1, -1, -1, -1, -1, -1, -1, -1, /* a0-a7 */
27 -1, -1, -1, -1, -1, -1, -1, -1, /* a8-af */
28 -1, -1, -1, -1, -1, -1, -1, -1, /* b0-b7 */
29 -1, -1, -1, -1, -1, -1, -1, -1, /* b8-bf */
30 -1, -1, -1, -1, -1, -1, -1, -1, /* c0-c7 */
31 -1, -1, -1, -1, -1, -1, -1, -1, /* c8-cf */
32 -1, -1, -1, -1, -1, -1, -1, -1, /* d0-d7 */
33 -1, -1, -1, -1, -1, -1, -1, -1, /* d8-df */
34 -1, -1, -1, -1, -1, -1, -1, -1, /* e0-e7 */
35 -1, -1, -1, -1, -1, -1, -1, -1, /* e8-ef */
36 -1, -1, -1, -1, -1, -1, -1, -1, /* f0-f7 */
37 -1, -1, -1, -1, -1, -1, -1, -1, /* f8-ff */
38};
39
René Scharfe0ec21862017-10-31 14:46:49 +010040int hex_to_bytes(unsigned char *binary, const char *hex, size_t len)
41{
42 for (; len; len--, hex += 2) {
43 unsigned int val = (hexval(hex[0]) << 4) | hexval(hex[1]);
44
45 if (val & ~0xff)
46 return -1;
47 *binary++ = val;
48 }
49 return 0;
50}
51
brian m. carlsondadacf12020-02-22 20:17:28 +000052static int get_hash_hex_algop(const char *hex, unsigned char *hash,
53 const struct git_hash_algo *algop)
Linus Torvaldsa5031212010-01-21 15:25:19 -080054{
55 int i;
brian m. carlsondadacf12020-02-22 20:17:28 +000056 for (i = 0; i < algop->rawsz; i++) {
René Scharfed2330972016-09-03 17:59:20 +020057 int val = hex2chr(hex);
58 if (val < 0)
Linus Torvaldsa5031212010-01-21 15:25:19 -080059 return -1;
brian m. carlsondadacf12020-02-22 20:17:28 +000060 *hash++ = val;
Linus Torvaldsa5031212010-01-21 15:25:19 -080061 hex += 2;
62 }
63 return 0;
64}
65
Junio C Hamano08e5fb12023-07-24 16:11:03 -070066int get_hash_hex(const char *hex, unsigned char *sha1)
brian m. carlsondadacf12020-02-22 20:17:28 +000067{
68 return get_hash_hex_algop(hex, sha1, the_hash_algo);
69}
70
71int get_oid_hex_algop(const char *hex, struct object_id *oid,
72 const struct git_hash_algo *algop)
73{
brian m. carlson5a6dce72021-04-26 01:02:55 +000074 int ret = get_hash_hex_algop(hex, oid->hash, algop);
75 if (!ret)
76 oid_set_algo(oid, algop);
77 return ret;
brian m. carlsondadacf12020-02-22 20:17:28 +000078}
79
brian m. carlson61e2a702020-02-22 20:17:29 +000080/*
81 * NOTE: This function relies on hash algorithms being in order from shortest
82 * length to longest length.
83 */
84int get_oid_hex_any(const char *hex, struct object_id *oid)
85{
86 int i;
87 for (i = GIT_HASH_NALGOS - 1; i > 0; i--) {
brian m. carlson5a6dce72021-04-26 01:02:55 +000088 if (!get_oid_hex_algop(hex, oid, &hash_algos[i]))
brian m. carlson61e2a702020-02-22 20:17:29 +000089 return i;
90 }
91 return GIT_HASH_UNKNOWN;
92}
93
brian m. carlsonaa1c6fd2015-03-13 23:39:28 +000094int get_oid_hex(const char *hex, struct object_id *oid)
95{
brian m. carlsondadacf12020-02-22 20:17:28 +000096 return get_oid_hex_algop(hex, oid, the_hash_algo);
97}
98
99int parse_oid_hex_algop(const char *hex, struct object_id *oid,
100 const char **end,
101 const struct git_hash_algo *algop)
102{
brian m. carlson5a6dce72021-04-26 01:02:55 +0000103 int ret = get_oid_hex_algop(hex, oid, algop);
brian m. carlsondadacf12020-02-22 20:17:28 +0000104 if (!ret)
105 *end = hex + algop->hexsz;
106 return ret;
brian m. carlsonaa1c6fd2015-03-13 23:39:28 +0000107}
108
brian m. carlson61e2a702020-02-22 20:17:29 +0000109int parse_oid_hex_any(const char *hex, struct object_id *oid, const char **end)
110{
111 int ret = get_oid_hex_any(hex, oid);
112 if (ret)
113 *end = hex + hash_algos[ret].hexsz;
114 return ret;
115}
116
brian m. carlson605f4302017-02-20 00:10:13 +0000117int parse_oid_hex(const char *hex, struct object_id *oid, const char **end)
118{
brian m. carlsondadacf12020-02-22 20:17:28 +0000119 return parse_oid_hex_algop(hex, oid, end, the_hash_algo);
brian m. carlson605f4302017-02-20 00:10:13 +0000120}
121
brian m. carlson47edb642018-11-14 04:09:29 +0000122char *hash_to_hex_algop_r(char *buffer, const unsigned char *hash,
123 const struct git_hash_algo *algop)
Linus Torvaldsa5031212010-01-21 15:25:19 -0800124{
Linus Torvaldsa5031212010-01-21 15:25:19 -0800125 static const char hex[] = "0123456789abcdef";
Jeff Kingaf49c6d2015-09-24 17:05:45 -0400126 char *buf = buffer;
Linus Torvaldsa5031212010-01-21 15:25:19 -0800127 int i;
128
brian m. carlsonb8505ec2021-04-26 01:03:00 +0000129 /*
130 * Our struct object_id has been memset to 0, so default to printing
131 * using the default hash.
132 */
133 if (algop == &hash_algos[0])
134 algop = the_hash_algo;
135
brian m. carlson47edb642018-11-14 04:09:29 +0000136 for (i = 0; i < algop->rawsz; i++) {
137 unsigned int val = *hash++;
Linus Torvaldsa5031212010-01-21 15:25:19 -0800138 *buf++ = hex[val >> 4];
139 *buf++ = hex[val & 0xf];
140 }
141 *buf = '\0';
142
143 return buffer;
144}
brian m. carlsonaa1c6fd2015-03-13 23:39:28 +0000145
brian m. carlson47edb642018-11-14 04:09:29 +0000146char *oid_to_hex_r(char *buffer, const struct object_id *oid)
147{
brian m. carlson3dd71462021-04-26 01:03:01 +0000148 return hash_to_hex_algop_r(buffer, oid->hash, &hash_algos[oid->algo]);
brian m. carlson47edb642018-11-14 04:09:29 +0000149}
150
151char *hash_to_hex_algop(const unsigned char *hash, const struct git_hash_algo *algop)
Jeff Kingaf49c6d2015-09-24 17:05:45 -0400152{
153 static int bufno;
brian m. carlsondc015052017-03-26 16:01:24 +0000154 static char hexbuffer[4][GIT_MAX_HEXSZ + 1];
René Scharfebb847352016-10-23 19:57:30 +0200155 bufno = (bufno + 1) % ARRAY_SIZE(hexbuffer);
brian m. carlson47edb642018-11-14 04:09:29 +0000156 return hash_to_hex_algop_r(hexbuffer[bufno], hash, algop);
157}
158
brian m. carlson47edb642018-11-14 04:09:29 +0000159char *hash_to_hex(const unsigned char *hash)
160{
161 return hash_to_hex_algop(hash, the_hash_algo);
Jeff Kingaf49c6d2015-09-24 17:05:45 -0400162}
163
brian m. carlsonaa1c6fd2015-03-13 23:39:28 +0000164char *oid_to_hex(const struct object_id *oid)
165{
brian m. carlson3dd71462021-04-26 01:03:01 +0000166 return hash_to_hex_algop(oid->hash, &hash_algos[oid->algo]);
brian m. carlsonaa1c6fd2015-03-13 23:39:28 +0000167}