1*8b6cd535SAndroid Build Coastguard Worker // Copyright (C) 2024 Google LLC
2*8b6cd535SAndroid Build Coastguard Worker //
3*8b6cd535SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*8b6cd535SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*8b6cd535SAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*8b6cd535SAndroid Build Coastguard Worker //
7*8b6cd535SAndroid Build Coastguard Worker // http://www.apache.org/licenses/LICENSE-2.0
8*8b6cd535SAndroid Build Coastguard Worker //
9*8b6cd535SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*8b6cd535SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*8b6cd535SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*8b6cd535SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*8b6cd535SAndroid Build Coastguard Worker // limitations under the License.
14*8b6cd535SAndroid Build Coastguard Worker
15*8b6cd535SAndroid Build Coastguard Worker #include "icing/util/sha256.h"
16*8b6cd535SAndroid Build Coastguard Worker
17*8b6cd535SAndroid Build Coastguard Worker #include <array>
18*8b6cd535SAndroid Build Coastguard Worker #include <cstdint>
19*8b6cd535SAndroid Build Coastguard Worker #include <cstring>
20*8b6cd535SAndroid Build Coastguard Worker
21*8b6cd535SAndroid Build Coastguard Worker namespace icing {
22*8b6cd535SAndroid Build Coastguard Worker namespace lib {
23*8b6cd535SAndroid Build Coastguard Worker
24*8b6cd535SAndroid Build Coastguard Worker // Constants for SHA-256 algorithm
25*8b6cd535SAndroid Build Coastguard Worker constexpr uint32_t k[64] = {
26*8b6cd535SAndroid Build Coastguard Worker 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1,
27*8b6cd535SAndroid Build Coastguard Worker 0x923f82a4, 0xab1c5ed5, 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3,
28*8b6cd535SAndroid Build Coastguard Worker 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, 0xe49b69c1, 0xefbe4786,
29*8b6cd535SAndroid Build Coastguard Worker 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da,
30*8b6cd535SAndroid Build Coastguard Worker 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147,
31*8b6cd535SAndroid Build Coastguard Worker 0x06ca6351, 0x14292967, 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13,
32*8b6cd535SAndroid Build Coastguard Worker 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, 0xa2bfe8a1, 0xa81a664b,
33*8b6cd535SAndroid Build Coastguard Worker 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070,
34*8b6cd535SAndroid Build Coastguard Worker 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a,
35*8b6cd535SAndroid Build Coastguard Worker 0x5b9cca4f, 0x682e6ff3, 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208,
36*8b6cd535SAndroid Build Coastguard Worker 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2};
37*8b6cd535SAndroid Build Coastguard Worker
38*8b6cd535SAndroid Build Coastguard Worker constexpr uint8_t kPaddingFirstByte = 0x80;
39*8b6cd535SAndroid Build Coastguard Worker constexpr uint8_t kNullChar = '\0';
40*8b6cd535SAndroid Build Coastguard Worker
41*8b6cd535SAndroid Build Coastguard Worker // Function to perform a right rotation on a 32-bit value
RightRotate(uint32_t value,unsigned int count)42*8b6cd535SAndroid Build Coastguard Worker uint32_t RightRotate(uint32_t value, unsigned int count) {
43*8b6cd535SAndroid Build Coastguard Worker return value >> count | value << (32 - count);
44*8b6cd535SAndroid Build Coastguard Worker }
45*8b6cd535SAndroid Build Coastguard Worker
Sha256()46*8b6cd535SAndroid Build Coastguard Worker Sha256::Sha256() {
47*8b6cd535SAndroid Build Coastguard Worker state_[0] = 0x6a09e667;
48*8b6cd535SAndroid Build Coastguard Worker state_[1] = 0xbb67ae85;
49*8b6cd535SAndroid Build Coastguard Worker state_[2] = 0x3c6ef372;
50*8b6cd535SAndroid Build Coastguard Worker state_[3] = 0xa54ff53a;
51*8b6cd535SAndroid Build Coastguard Worker state_[4] = 0x510e527f;
52*8b6cd535SAndroid Build Coastguard Worker state_[5] = 0x9b05688c;
53*8b6cd535SAndroid Build Coastguard Worker state_[6] = 0x1f83d9ab;
54*8b6cd535SAndroid Build Coastguard Worker state_[7] = 0x5be0cd19;
55*8b6cd535SAndroid Build Coastguard Worker count_ = 0;
56*8b6cd535SAndroid Build Coastguard Worker memset(buffer_.data(), 0, sizeof(buffer_));
57*8b6cd535SAndroid Build Coastguard Worker }
58*8b6cd535SAndroid Build Coastguard Worker
Transform()59*8b6cd535SAndroid Build Coastguard Worker void Sha256::Transform() {
60*8b6cd535SAndroid Build Coastguard Worker uint32_t w[64];
61*8b6cd535SAndroid Build Coastguard Worker int t = 0;
62*8b6cd535SAndroid Build Coastguard Worker // Process the first 16 words of the message block
63*8b6cd535SAndroid Build Coastguard Worker for (; t < 16; ++t) {
64*8b6cd535SAndroid Build Coastguard Worker uint32_t tmp = static_cast<uint32_t>(buffer_[t * 4]) << 24;
65*8b6cd535SAndroid Build Coastguard Worker tmp |= static_cast<uint32_t>(buffer_[t * 4 + 1]) << 16;
66*8b6cd535SAndroid Build Coastguard Worker tmp |= static_cast<uint32_t>(buffer_[t * 4 + 2]) << 8;
67*8b6cd535SAndroid Build Coastguard Worker tmp |= static_cast<uint32_t>(buffer_[t * 4 + 3]);
68*8b6cd535SAndroid Build Coastguard Worker w[t] = tmp;
69*8b6cd535SAndroid Build Coastguard Worker }
70*8b6cd535SAndroid Build Coastguard Worker
71*8b6cd535SAndroid Build Coastguard Worker // Extend the first 16 words into the remaining 48 words of the message
72*8b6cd535SAndroid Build Coastguard Worker // schedule
73*8b6cd535SAndroid Build Coastguard Worker for (; t < 64; t++) {
74*8b6cd535SAndroid Build Coastguard Worker // Calculate the next word in the message schedule based on the previous
75*8b6cd535SAndroid Build Coastguard Worker // words
76*8b6cd535SAndroid Build Coastguard Worker uint32_t s0 = RightRotate(w[t - 15], 7) ^ RightRotate(w[t - 15], 18) ^
77*8b6cd535SAndroid Build Coastguard Worker (w[t - 15] >> 3);
78*8b6cd535SAndroid Build Coastguard Worker uint32_t s1 = RightRotate(w[t - 2], 17) ^ RightRotate(w[t - 2], 19) ^
79*8b6cd535SAndroid Build Coastguard Worker (w[t - 2] >> 10);
80*8b6cd535SAndroid Build Coastguard Worker w[t] = w[t - 16] + s0 + w[t - 7] + s1;
81*8b6cd535SAndroid Build Coastguard Worker }
82*8b6cd535SAndroid Build Coastguard Worker
83*8b6cd535SAndroid Build Coastguard Worker uint32_t a = state_[0];
84*8b6cd535SAndroid Build Coastguard Worker uint32_t b = state_[1];
85*8b6cd535SAndroid Build Coastguard Worker uint32_t c = state_[2];
86*8b6cd535SAndroid Build Coastguard Worker uint32_t d = state_[3];
87*8b6cd535SAndroid Build Coastguard Worker uint32_t e = state_[4];
88*8b6cd535SAndroid Build Coastguard Worker uint32_t f = state_[5];
89*8b6cd535SAndroid Build Coastguard Worker uint32_t g = state_[6];
90*8b6cd535SAndroid Build Coastguard Worker uint32_t h = state_[7];
91*8b6cd535SAndroid Build Coastguard Worker
92*8b6cd535SAndroid Build Coastguard Worker for (int i = 0; i < 64; i++) {
93*8b6cd535SAndroid Build Coastguard Worker uint32_t sigma0 =
94*8b6cd535SAndroid Build Coastguard Worker RightRotate(a, 2) ^ RightRotate(a, 13) ^ RightRotate(a, 22);
95*8b6cd535SAndroid Build Coastguard Worker uint32_t majority = (a & b) ^ (a & c) ^ (b & c);
96*8b6cd535SAndroid Build Coastguard Worker uint32_t temp2 = sigma0 + majority;
97*8b6cd535SAndroid Build Coastguard Worker uint32_t sigma1 =
98*8b6cd535SAndroid Build Coastguard Worker RightRotate(e, 6) ^ RightRotate(e, 11) ^ RightRotate(e, 25);
99*8b6cd535SAndroid Build Coastguard Worker uint32_t choice = (e & f) ^ ((~e) & g);
100*8b6cd535SAndroid Build Coastguard Worker uint32_t temp1 = h + sigma1 + choice + k[i] + w[i];
101*8b6cd535SAndroid Build Coastguard Worker
102*8b6cd535SAndroid Build Coastguard Worker h = g;
103*8b6cd535SAndroid Build Coastguard Worker g = f;
104*8b6cd535SAndroid Build Coastguard Worker f = e;
105*8b6cd535SAndroid Build Coastguard Worker e = d + temp1;
106*8b6cd535SAndroid Build Coastguard Worker d = c;
107*8b6cd535SAndroid Build Coastguard Worker c = b;
108*8b6cd535SAndroid Build Coastguard Worker b = a;
109*8b6cd535SAndroid Build Coastguard Worker a = temp1 + temp2;
110*8b6cd535SAndroid Build Coastguard Worker }
111*8b6cd535SAndroid Build Coastguard Worker
112*8b6cd535SAndroid Build Coastguard Worker state_[0] += a;
113*8b6cd535SAndroid Build Coastguard Worker state_[1] += b;
114*8b6cd535SAndroid Build Coastguard Worker state_[2] += c;
115*8b6cd535SAndroid Build Coastguard Worker state_[3] += d;
116*8b6cd535SAndroid Build Coastguard Worker state_[4] += e;
117*8b6cd535SAndroid Build Coastguard Worker state_[5] += f;
118*8b6cd535SAndroid Build Coastguard Worker state_[6] += g;
119*8b6cd535SAndroid Build Coastguard Worker state_[7] += h;
120*8b6cd535SAndroid Build Coastguard Worker }
121*8b6cd535SAndroid Build Coastguard Worker
Update(const uint8_t * data,size_t length)122*8b6cd535SAndroid Build Coastguard Worker void Sha256::Update(const uint8_t* data, size_t length) {
123*8b6cd535SAndroid Build Coastguard Worker int i = static_cast<int>(count_ & 0b111111);
124*8b6cd535SAndroid Build Coastguard Worker count_ += length;
125*8b6cd535SAndroid Build Coastguard Worker while (length--) {
126*8b6cd535SAndroid Build Coastguard Worker buffer_[i] = *data;
127*8b6cd535SAndroid Build Coastguard Worker ++data;
128*8b6cd535SAndroid Build Coastguard Worker ++i;
129*8b6cd535SAndroid Build Coastguard Worker if (i == 64) {
130*8b6cd535SAndroid Build Coastguard Worker Transform();
131*8b6cd535SAndroid Build Coastguard Worker i = 0;
132*8b6cd535SAndroid Build Coastguard Worker }
133*8b6cd535SAndroid Build Coastguard Worker }
134*8b6cd535SAndroid Build Coastguard Worker }
135*8b6cd535SAndroid Build Coastguard Worker
Finalize()136*8b6cd535SAndroid Build Coastguard Worker std::array<uint8_t, 32> Sha256::Finalize() && {
137*8b6cd535SAndroid Build Coastguard Worker uint64_t bits_count = count_ << 3;
138*8b6cd535SAndroid Build Coastguard Worker
139*8b6cd535SAndroid Build Coastguard Worker // SHA-256 padding: the message is padded with a '1' bit, then with '0' bits
140*8b6cd535SAndroid Build Coastguard Worker // until the message length modulo 512 is 448 bits (56 bytes modulo 64).
141*8b6cd535SAndroid Build Coastguard Worker Update(&kPaddingFirstByte, 1);
142*8b6cd535SAndroid Build Coastguard Worker
143*8b6cd535SAndroid Build Coastguard Worker // Pad with '0' bits until the message length modulo 64 is 56 bytes, leaving
144*8b6cd535SAndroid Build Coastguard Worker // 8 bytes for the length of the original message.
145*8b6cd535SAndroid Build Coastguard Worker while (count_ % 64 != 56) {
146*8b6cd535SAndroid Build Coastguard Worker Update(&kNullChar, 1);
147*8b6cd535SAndroid Build Coastguard Worker }
148*8b6cd535SAndroid Build Coastguard Worker
149*8b6cd535SAndroid Build Coastguard Worker // Append the length of the original message (in bits) to the end of the
150*8b6cd535SAndroid Build Coastguard Worker // padded message in big-endian order.
151*8b6cd535SAndroid Build Coastguard Worker for (int i = 0; i < 8; ++i) {
152*8b6cd535SAndroid Build Coastguard Worker uint8_t tmp = static_cast<uint8_t>(bits_count >> 56);
153*8b6cd535SAndroid Build Coastguard Worker bits_count <<= 8;
154*8b6cd535SAndroid Build Coastguard Worker Update(&tmp, 1);
155*8b6cd535SAndroid Build Coastguard Worker }
156*8b6cd535SAndroid Build Coastguard Worker
157*8b6cd535SAndroid Build Coastguard Worker // Convert the state array to a 32-byte sha256 hash array in big-endian order.
158*8b6cd535SAndroid Build Coastguard Worker std::array<uint8_t, 32> hash;
159*8b6cd535SAndroid Build Coastguard Worker for (int i = 0, j = 0; i < 8; i++) {
160*8b6cd535SAndroid Build Coastguard Worker uint32_t tmp = state_[i];
161*8b6cd535SAndroid Build Coastguard Worker hash[j++] = static_cast<uint8_t>(tmp >> 24);
162*8b6cd535SAndroid Build Coastguard Worker hash[j++] = static_cast<uint8_t>(tmp >> 16);
163*8b6cd535SAndroid Build Coastguard Worker hash[j++] = static_cast<uint8_t>(tmp >> 8);
164*8b6cd535SAndroid Build Coastguard Worker hash[j++] = static_cast<uint8_t>(tmp);
165*8b6cd535SAndroid Build Coastguard Worker }
166*8b6cd535SAndroid Build Coastguard Worker
167*8b6cd535SAndroid Build Coastguard Worker return hash;
168*8b6cd535SAndroid Build Coastguard Worker }
169*8b6cd535SAndroid Build Coastguard Worker
170*8b6cd535SAndroid Build Coastguard Worker } // namespace lib
171*8b6cd535SAndroid Build Coastguard Worker } // namespace icing
172