aboutsummaryrefslogtreecommitdiff
path: root/evc/src/bin/encode.rs
blob: 43f2c57d2df5703e30fc9ffa5b9bc555e3654a67 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
use anyhow::Context;
use clap::Parser;
use indicatif::ProgressBar;
use libreschmux::{
    codec::{
        decode::{decode_block, DecodeConfig},
        encode::{encode_block, EncodeConfig, EncodeMode},
    },
    format::{header::Header, ser::Sink},
    frame::Frame,
    helpers::vector::Vec2,
};
use log::info;
use std::io::{BufReader, BufWriter};

#[derive(Parser)]
#[clap(about, version)]
pub struct EncodeArgs {
    #[arg(short = 'W', long)]
    width: usize,
    #[arg(short = 'H', long)]
    height: usize,

    #[arg(short, long)]
    mode: EncodeMode,

    #[arg(long)]
    no_linear_transform: bool,
    #[arg(long)]
    no_value_scale: bool,
    #[arg(long)]
    no_translate: bool,

    #[arg[short = 'L', long]]
    length: Option<usize>,

    #[arg(short, long, default_value = "8")]
    min_block_size: isize,

    #[arg(short = 't', long, default_value = "200")]
    ref_thres: f64,
}

fn main() -> anyhow::Result<()> {
    env_logger::init_from_env("LOG");
    let args = EncodeArgs::parse();

    let mut input = BufReader::new(std::io::stdin());
    let mut output = BufWriter::new(std::io::stdout());

    let config = EncodeConfig {
        mode: args.mode,
        ref_thres: args.ref_thres,
        weight_factor: 50.0,
        max_diff_area: 10_000,
        min_block_size: args.min_block_size,
        do_translate: !args.no_translate,
        do_linear_transform: !args.no_linear_transform,
        do_value_scale: !args.no_value_scale,
    };

    let size = Vec2 {
        x: args.width as isize,
        y: args.height as isize,
    };
    output
        .put(Header {
            resolution: size,
            frame_count: args.length.unwrap_or(usize::MAX),
        })
        .context("writing header")?;

    let progress_bar = args.length.map(|len| ProgressBar::new(len as u64));
    let mut prev_frame = Frame::new(size);

    for i in 0.. {
        info!("encode frame {i}");
        let mut frame = Frame::read(&mut input, size).context("reading raw frame")?;

        let v1 = frame.view();
        let v2 = prev_frame.view();

        let (error, root) = encode_block(v1, v2, &config);

        // compress_block(
        //     &mut root,
        //     Vec2 {
        //         x: size.x as usize,
        //         y: size.y as usize,
        //     },
        // );

        root.write(&mut output, size)
            .context("writing encoded frame")?;

        info!(
            "cumulative error: {error} ({} per pixel)",
            error / frame.view().area() as f64
        );

        if let Some(progress_bar) = &progress_bar {
            progress_bar.inc(1);
        }

        decode_block(&root, frame.view_mut(), prev_frame.view(), &DecodeConfig {});

        prev_frame = frame;
    }

    Ok(())
}