/ref/c++/segwit_addr.cpp
https://github.com/sipa/bech32 · C++ · 86 lines · 53 code · 10 blank · 23 comment · 27 complexity · 4ea8e5c1fc5804c12e3882c3fa23ca9c MD5 · raw file
- /* Copyright (c) 2017, 2021 Pieter Wuille
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- * THE SOFTWARE.
- */
- #include "segwit_addr.h"
- #include "bech32.h"
- #include <tuple>
- namespace
- {
- typedef std::vector<uint8_t> data;
- /** Convert from one power-of-2 number base to another. */
- template<int frombits, int tobits, bool pad>
- bool convertbits(data& out, const data& in) {
- int acc = 0;
- int bits = 0;
- const int maxv = (1 << tobits) - 1;
- const int max_acc = (1 << (frombits + tobits - 1)) - 1;
- for (size_t i = 0; i < in.size(); ++i) {
- int value = in[i];
- acc = ((acc << frombits) | value) & max_acc;
- bits += frombits;
- while (bits >= tobits) {
- bits -= tobits;
- out.push_back((acc >> bits) & maxv);
- }
- }
- if (pad) {
- if (bits) out.push_back((acc << (tobits - bits)) & maxv);
- } else if (bits >= frombits || ((acc << (tobits - bits)) & maxv)) {
- return false;
- }
- return true;
- }
- }
- namespace segwit_addr
- {
- /** Decode a SegWit address. */
- std::pair<int, data> decode(const std::string& hrp, const std::string& addr) {
- const auto dec = bech32::decode(addr);
- if (dec.hrp != hrp || dec.data.size() < 1) return std::make_pair(-1, data());
- data conv;
- uint8_t witver = dec.data[0];
- if (!convertbits<5, 8, false>(conv, data(dec.data.begin() + 1, dec.data.end())) ||
- conv.size() < 2 || conv.size() > 40 || witver > 16 || (witver == 0 &&
- conv.size() != 20 && conv.size() != 32) || (witver == 0 && dec.encoding !=
- bech32::Encoding::BECH32) || (witver != 0 && dec.encoding != bech32::Encoding::BECH32M)) {
- return std::make_pair(-1, data());
- }
- return std::make_pair(dec.data[0], conv);
- }
- /** Encode a SegWit address. */
- std::string encode(const std::string& hrp, int witver, const data& witprog) {
- data enc;
- enc.push_back(witver);
- convertbits<8, 5, true>(enc, witprog);
- std::string ret = bech32::encode(hrp, enc, witver > 0 ? bech32::Encoding::BECH32M : bech32::Encoding::BECH32);
- if (decode(hrp, ret).first == -1) return "";
- return ret;
- }
- }