跳转至

File cipher_rc4.cc

File List > crypto > cipher_rc4.cc

Go to the documentation of this file

#include <qqmusic/crypto/cipher_rc4.h>

namespace qqmusic::crypto {
RC4Cipher::RC4Cipher(const std::vector<uint8_t>& key)
    : key(key) {
    if (key.size() == 0) {
        std::cout << "[RC4Cipher constructor failed] key size is 0" << std::endl;
    }
}

void RC4Cipher::decrypt(qqmusic::utils::buffer& buf, size_t offset) {
    // 初始化S盒
    std::vector<uint8_t> box(key.size());
    std::iota(box.begin(), box.end(), 0);

    // KSA密钥调度算法
    size_t j = 0;
    for (size_t i = 0; i < key.size(); ++i) {
        j = (j + box[i] + key[i % key.size()]) % key.size();
        std::swap(box[i], box[j]);
    }

    // 哈希值计算
    uint32_t hash_base = 1;
    for (auto k : key) {
        if (k == 0)
            continue;
        const auto next_hash = hash_base * k;
        if (next_hash == 0 || next_hash <= hash_base)
            break;
        hash_base = next_hash;
    }

    // 解密主逻辑
    constexpr size_t SegmentSize = 5120;
    constexpr size_t FirstSegmentSize = 128;

    auto enc_first = [&](auto& buf, auto len, auto off) {
        for (size_t i = 0; i < len; ++i) {
            const auto seg_id = (off + i) / SegmentSize;
            const auto seed = key[seg_id % key.size()];
            const auto idx = static_cast<double>(hash_base) / ((seg_id + 1) * seed) * 100.0;
            const auto skip = static_cast<size_t>(idx) % key.size();
            buf[i] ^= key[skip];
        }
    };

    auto enc_regular = [&](auto& buf, auto len, auto off) {
        auto local_box = box;
        int j = 0, k = 0;
        const int seg_num = off / SegmentSize;
        const int seed = key[seg_num % key.size()];
        const int idx = static_cast<double>(hash_base) / ((seg_num + 1) * seed) * 100.0;
        const int skip_len = (off % SegmentSize) + (static_cast<size_t>(idx) % key.size());

        for (int i = -static_cast<int>(skip_len); i < static_cast<int>(len); ++i) {
            j = (j + i) % static_cast<int>(key.size());
            k = (local_box[j] + k) % static_cast<int>(key.size());
            std::swap(local_box[j], local_box[k]);
            if (i >= 0) {
                buf[i] ^= local_box[(local_box[j] + local_box[k]) % key.size()];
            }
        }
    };
    auto process_segment = [&](size_t block_size, auto enc_func) -> size_t {
        const size_t chunk = std::min(block_size, buf.size());
        if (chunk == 0)
            return true;

        enc_func(buf, chunk, offset);
        return chunk;
    };

    size_t total_processed = 0;
    const size_t total_size = buf.size();

    while (total_processed < total_size) {
        size_t processed = 0;

        // 处理第一段
        if (offset < FirstSegmentSize) {
            const size_t remaining = FirstSegmentSize - offset;
            processed = process_segment(remaining, enc_first);
            total_processed += processed;
            offset += processed;
            if (processed < remaining)
                break; // 数据不足时退出
        }

        // 处理常规段
        const size_t remaining = total_size - total_processed;
        processed = process_segment(std::min(remaining, SegmentSize), enc_regular);
        total_processed += processed;
        offset += processed;
    }
}

} // namespace qqmusic::crypto