2019-11-04 09:55:43 -05:00
|
|
|
#include <qpdf/Pl_SHA2.hh>
|
2022-02-04 16:31:31 -05:00
|
|
|
|
2019-11-04 09:55:43 -05:00
|
|
|
#include <qpdf/QPDFCryptoProvider.hh>
|
2022-04-02 17:14:10 -04:00
|
|
|
#include <qpdf/QUtil.hh>
|
|
|
|
#include <stdexcept>
|
2019-11-04 09:55:43 -05:00
|
|
|
|
|
|
|
Pl_SHA2::Pl_SHA2(int bits, Pipeline* next) :
|
|
|
|
Pipeline("sha2", next),
|
|
|
|
in_progress(false)
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (bits) {
|
2019-11-04 09:55:43 -05:00
|
|
|
resetBits(bits);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2022-05-03 17:43:07 -04:00
|
|
|
Pl_SHA2::write(unsigned char const* buf, size_t len)
|
2019-11-04 09:55:43 -05:00
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (!this->in_progress) {
|
2022-02-08 09:18:08 -05:00
|
|
|
this->in_progress = true;
|
2019-11-04 09:55:43 -05:00
|
|
|
}
|
|
|
|
|
2023-05-27 18:19:52 +01:00
|
|
|
// Write in chunks in case len is too big to fit in an int. Assume int is at least 32 bits.
|
2019-11-04 09:55:43 -05:00
|
|
|
static size_t const max_bytes = 1 << 30;
|
|
|
|
size_t bytes_left = len;
|
2022-05-03 17:43:07 -04:00
|
|
|
unsigned char const* data = buf;
|
2022-04-02 17:14:10 -04:00
|
|
|
while (bytes_left > 0) {
|
2022-02-08 09:18:08 -05:00
|
|
|
size_t bytes = (bytes_left >= max_bytes ? max_bytes : bytes_left);
|
2019-11-04 09:55:43 -05:00
|
|
|
this->crypto->SHA2_update(data, bytes);
|
2022-02-08 09:18:08 -05:00
|
|
|
bytes_left -= bytes;
|
2019-11-04 09:55:43 -05:00
|
|
|
data += bytes;
|
|
|
|
}
|
|
|
|
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->getNext(true)) {
|
2019-11-04 09:55:43 -05:00
|
|
|
this->getNext()->write(buf, len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Pl_SHA2::finish()
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->getNext(true)) {
|
2019-11-04 09:55:43 -05:00
|
|
|
this->getNext()->finish();
|
|
|
|
}
|
|
|
|
this->crypto->SHA2_finalize();
|
|
|
|
this->in_progress = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Pl_SHA2::resetBits(int bits)
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->in_progress) {
|
2023-05-21 13:35:09 -04:00
|
|
|
throw std::logic_error("bit reset requested for in-progress SHA2 Pipeline");
|
2019-11-04 09:55:43 -05:00
|
|
|
}
|
|
|
|
this->crypto = QPDFCryptoProvider::getImpl();
|
|
|
|
this->crypto->SHA2_init(bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
Pl_SHA2::getRawDigest()
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->in_progress) {
|
2023-05-21 13:35:09 -04:00
|
|
|
throw std::logic_error("digest requested for in-progress SHA2 Pipeline");
|
2019-11-04 09:55:43 -05:00
|
|
|
}
|
|
|
|
return this->crypto->SHA2_digest();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
Pl_SHA2::getHexDigest()
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (this->in_progress) {
|
2023-05-21 13:35:09 -04:00
|
|
|
throw std::logic_error("digest requested for in-progress SHA2 Pipeline");
|
2019-11-04 09:55:43 -05:00
|
|
|
}
|
|
|
|
return QUtil::hex_encode(getRawDigest());
|
|
|
|
}
|