add artemis archive pack support

This commit is contained in:
2025-07-22 21:51:15 +08:00
parent f4b10051a1
commit 80101328c0
4 changed files with 203 additions and 1 deletions

View File

@@ -157,6 +157,10 @@ pub struct Arg {
#[arg(long, global = true)]
/// Whether to overlay PIMG images. (By default, true if all layers are not group layers.)
pub kirikiri_pimg_overlay: Option<bool>,
#[cfg(feature = "artemis-arc")]
#[arg(long, global = true)]
/// Disable Artemis archive (.pfs) XOR encryption when packing.
pub artemis_arc_disable_xor: bool,
#[command(subcommand)]
/// Command
pub command: Command,

View File

@@ -1432,6 +1432,8 @@ fn main() {
bgi_compress_file: arg.bgi_compress_file,
#[cfg(feature = "kirikiri-img")]
kirikiri_pimg_overlay: arg.kirikiri_pimg_overlay,
#[cfg(feature = "artemis-arc")]
artemis_arc_disable_xor: arg.artemis_arc_disable_xor,
};
match &arg.command {
args::Command::Export { input, output } => {

View File

@@ -5,6 +5,7 @@ use crate::utils::struct_pack::*;
use anyhow::Result;
use msg_tool_macro::*;
use sha1::Digest;
use std::collections::HashMap;
use std::io::{Read, Seek, SeekFrom, Write};
use std::sync::{Arc, Mutex};
@@ -86,6 +87,29 @@ impl ScriptBuilder for ArtemisArcBuilder {
fn script_type(&self) -> &'static ScriptType {
&ScriptType::ArtemisArc
}
fn is_this_format(&self, _filename: &str, buf: &[u8], buf_len: usize) -> Option<u8> {
if buf_len >= 3 && (buf.starts_with(b"pf6") || buf.starts_with(b"pf8")) {
return Some(10);
}
None
}
fn create_archive(
&self,
filename: &str,
files: &[&str],
encoding: Encoding,
config: &ExtraConfig,
) -> Result<Box<dyn Archive>> {
let f = std::fs::File::options()
.write(true)
.read(true)
.create(true)
.truncate(true)
.open(filename)?;
Ok(Box::new(ArtemisArcWriter::new(f, files, encoding, config)?))
}
}
#[derive(Debug, Clone, StructPack, StructUnpack)]
@@ -121,7 +145,7 @@ impl<T: Read + Seek + std::fmt::Debug> ArtemisArc<T> {
));
}
let version = reader.read_u8()?;
if version != b'2' && version != b'6' && version != b'8' {
if version != b'6' && version != b'8' {
return Err(anyhow::anyhow!(
"Unsupported Artemis archive version: {}",
version
@@ -304,3 +328,173 @@ impl<'a, T: Iterator<Item = &'a PfsEntryHeader>, R: Read + Seek + 'static> Itera
}
}
}
pub struct ArtemisArcWriter<T: Write + Seek + Read> {
writer: T,
headers: HashMap<String, PfsEntryHeader>,
encoding: Encoding,
disable_xor: bool,
index_size: u32,
}
impl<T: Write + Seek + Read> ArtemisArcWriter<T> {
pub fn new(
mut writer: T,
files: &[&str],
encoding: Encoding,
config: &ExtraConfig,
) -> Result<Self> {
writer.write_all(if config.artemis_arc_disable_xor {
b"pf6"
} else {
b"pf8"
})?;
writer.write_u32(0)?; // Placeholder for index size
writer.write_u32(files.len() as u32)?;
let mut headers = HashMap::new();
for file in files {
let header = PfsEntryHeader {
name: file.to_string(),
_unk: 0,
offset: 0,
size: 0,
};
header.pack(&mut writer, false, encoding)?;
headers.insert(file.to_string(), header);
}
let size = writer.stream_position()?;
let index_size = size as u32 - 7;
writer.write_u32_at(3, index_size)?;
Ok(ArtemisArcWriter {
writer,
headers,
encoding,
disable_xor: config.artemis_arc_disable_xor,
index_size,
})
}
}
impl<T: Write + Seek + Read> Archive for ArtemisArcWriter<T> {
fn new_file<'a>(&'a mut self, name: &str) -> Result<Box<dyn WriteSeek + 'a>> {
let entry = self
.headers
.get_mut(name)
.ok_or_else(|| anyhow::anyhow!("File '{}' not found in archive", name))?;
if entry.offset != 0 || entry.size != 0 {
return Err(anyhow::anyhow!("File '{}' already exists in archive", name));
}
self.writer.seek(SeekFrom::End(0))?;
entry.offset = self.writer.stream_position()? as u32;
let file = ArtemisArcFile {
header: entry,
writer: &mut self.writer,
pos: 0,
};
Ok(Box::new(file))
}
fn write_header(&mut self) -> Result<()> {
self.writer.seek(SeekFrom::Start(11))?;
let mut files = self.headers.values().collect::<Vec<_>>();
files.sort_by_key(|d| d.offset);
for file in files.iter() {
file.pack(&mut self.writer, false, self.encoding)?;
}
if !self.disable_xor {
self.writer.seek(SeekFrom::Start(7))?;
let mut sha = sha1::Sha1::default();
let w = &mut self.writer;
let mut header = w.take(self.index_size as u64);
std::io::copy(&mut header, &mut sha)?;
sha.flush()?;
let result = sha.finalize();
let mut xor_key = [0u8; 20];
xor_key.copy_from_slice(&result);
let mut buf = [0u8; 1024];
for file in files.iter() {
self.writer.seek(SeekFrom::Start(file.offset as u64))?;
let mut pos = 0u32;
while pos < file.size {
let bytes_to_read = (file.size - pos).min(1024) as usize;
let bytes_read = self.writer.read(&mut buf[..bytes_to_read])?;
if bytes_read == 0 {
return Err(anyhow::anyhow!(
"Unexpected end of file while reading '{}'",
file.name
));
}
for i in 0..bytes_read {
let l = (pos as u64 + i as u64) % 20;
buf[i] ^= xor_key[l as usize];
}
self.writer.seek_relative(-(bytes_read as i64))?;
self.writer.write_all(&buf[..bytes_read])?;
pos += bytes_read as u32;
}
}
}
Ok(())
}
}
pub struct ArtemisArcFile<'a, T: Write + Seek> {
header: &'a mut PfsEntryHeader,
writer: &'a mut T,
pos: u64,
}
impl<'a, T: Write + Seek> Write for ArtemisArcFile<'a, T> {
fn write(&mut self, buf: &[u8]) -> std::io::Result<usize> {
self.writer
.seek(SeekFrom::Start(self.header.offset as u64 + self.pos))?;
let bytes_written = self.writer.write(buf)?;
self.pos += bytes_written as u64;
self.header.size = self.header.size.max(self.pos as u32);
Ok(bytes_written)
}
fn flush(&mut self) -> std::io::Result<()> {
self.writer.flush()
}
}
impl<'a, T: Write + Seek> Seek for ArtemisArcFile<'a, T> {
fn seek(&mut self, pos: SeekFrom) -> std::io::Result<u64> {
let new_pos = match pos {
SeekFrom::Start(offset) => offset,
SeekFrom::End(offset) => {
if offset < 0 {
if (-offset) as u64 > self.header.size as u64 {
return Err(std::io::Error::new(
std::io::ErrorKind::InvalidInput,
"Seek from end exceeds file length",
));
}
self.header.size as u64 - (-offset) as u64
} else {
self.header.size as u64 + offset as u64
}
}
SeekFrom::Current(offset) => {
if offset < 0 {
if (-offset) as u64 > self.pos {
return Err(std::io::Error::new(
std::io::ErrorKind::InvalidInput,
"Seek from current exceeds current position",
));
}
self.pos.saturating_sub((-offset) as u64)
} else {
self.pos + offset as u64
}
}
};
self.pos = new_pos;
Ok(self.pos)
}
fn stream_position(&mut self) -> std::io::Result<u64> {
Ok(self.pos)
}
}

View File

@@ -223,6 +223,8 @@ pub struct ExtraConfig {
pub bgi_compress_file: bool,
#[cfg(feature = "kirikiri-img")]
pub kirikiri_pimg_overlay: Option<bool>,
#[cfg(feature = "artemis-arc")]
pub artemis_arc_disable_xor: bool,
}
#[derive(Clone, Copy, Debug, ValueEnum, PartialEq, Eq, PartialOrd, Ord)]