Crate blend_info
source · [−]Expand description
blend_info
Print some information about a Blender scene file.
$ ./target/release/blend_info -h
blend_info 0.2.9
Print some information about a Blender scene file
USAGE:
blend_info [FLAGS] [OPTIONS] <path>
FLAGS:
--dna Print information about DNA of Blender
-h, --help Prints help information
--pointers Print code (e.g. OB, CA, LA, MA, DATA) and pointers
-V, --version Prints version information
OPTIONS:
-n, --struct_name <struct-name> Print information about a particular struct
ARGS:
<path> The path to the file to read
Usage as a crate (used in your own code)
use blend_info::{print_pointer, read_dna, use_dna, DnaStrC};
// std
use std::collections::HashMap;
use std::path::PathBuf;
fn main() -> std::io::Result<()> {
println!("use_dna");
// .blend file
let path: PathBuf = PathBuf::from(r"blend/factory_v279.blend");
println!("Try to read {:?} file ...", path);
// read DNA
let mut dna_types_hm: HashMap<String, u16> = HashMap::new();
let mut dna_structs_hm: HashMap<String, DnaStrC> = HashMap::new();
let mut dna_pointers_hm: HashMap<usize, usize> = HashMap::new();
let mut dna_2_type_id: Vec<u16> = Vec::new();
let mut types: Vec<String> = Vec::new();
let mut num_bytes_read: usize = 0;
let print_dna: bool = false;
let print_pointers: bool = true;
read_dna(
print_dna,
print_pointers,
&path,
&mut dna_types_hm,
&mut dna_structs_hm,
&mut dna_pointers_hm,
&mut dna_2_type_id,
&mut types,
&mut num_bytes_read,
)?;
println!("{} bytes read by read_dna() ...", num_bytes_read);
// use DNA
println!(
"Try to read {:?} file again (camera/lamp centric this time) ...",
path
);
let print_dna: bool = true;
let names: Vec<String> = vec!["Camera".to_string(), "Lamp".to_string()];
let mut bytes_read: Vec<u8> = Vec::with_capacity(num_bytes_read);
let mut structs_read: Vec<String> = Vec::with_capacity(names.len());
let mut data_read: Vec<u32> = Vec::with_capacity(names.len());
let mut pointers_read: Vec<(usize, u32)> = Vec::with_capacity(names.len());
use_dna(
print_dna,
&path,
&dna_types_hm,
&dna_structs_hm,
&names,
&dna_2_type_id,
&types,
&mut bytes_read,
&mut structs_read,
&mut data_read,
&mut pointers_read,
)?;
println!("bytes_read: {:?}", bytes_read);
println!("structs_read: {:?}", structs_read);
println!("data_read: {:?}", data_read);
println!("pointers_read: {:?}", pointers_read);
print!("SDNAnr = {}: ", pointers_read[0].1);
print_pointer(pointers_read[0].0, &names[0], &dna_pointers_hm);
print!("SDNAnr = {}: ", pointers_read[1].1);
print_pointer(pointers_read[1].0, &names[1], &dna_pointers_hm);
Ok(())
}
Examples (used as a standalone executable to query .blend files)
DNA
Find Blender version and read all bytes:
$ ./target/release/blend_info --dna blend/factory_v279.blend | less
BLENDER-v279
...
459792 bytes read
$ ./target/release/blend_info --dna blend/factory_v300.blend | less
BLENDER-v300
...
806388 bytes read
Get an idea what structs might be useful and which names are defined:
$ ./target/release/blend_info --dna blend/factory_v279.blend | grep "\[SDNAnr =" -A 1
[SDNAnr = 0]
Link (len=16) {
--
[SDNAnr = 1]
LinkData (len=24) {
...
--
[SDNAnr = 620]
CacheFile (len=1200) {
Structs and their contained data
$ ./target/release/blend_info -n Camera blend/factory_v279.blend
Camera 248
struct Camera { // SDNAnr = 25
ID id; // 120
AnimData *adt; // 8
char type; // 1
char dtx; // 1
short flag; // 2
float passepartalpha; // 4
float clipsta; // 4
float clipend; // 4
float lens; // 4
float ortho_scale; // 4
float drawsize; // 4
float sensor_x; // 4
float sensor_y; // 4
float shiftx; // 4
float shifty; // 4
float YF_dofdist; // 4
Ipo *ipo; // 8
Object *dof_ob; // 8
GPUDOFSettings gpu_dof; // 24
char sensor_fit; // 1
char pad[7]; // 7
CameraStereoSettings stereo; // 24
}; // 248
$ ./target/release/blend_info -n ID blend/factory_v279.blend
ID 120
struct ID { // SDNAnr = 10
void *next; // 8
void *prev; // 8
ID *newid; // 8
Library *lib; // 8
char name[66]; // 66
short flag; // 2
short tag; // 2
short pad_s1; // 2
int us; // 4
int icon_id; // 4
IDProperty *properties; // 8
}; // 120
$ ./target/release/blend_info -n Object.id.name blend/factory_v279.blend
Object.id.name = "OBCamera"
Object.id.name = "OBCube"
Object.id.name = "OBLamp"
$ ./target/release/blend_info -n Camera.id.name blend/factory_v279.blend
Camera.id.name = "CACamera"
$ ./target/release/blend_info -n Lamp.id.name blend/factory_v279.blend
Lamp.id.name = "LALamp"
$ ./target/release/blend_info -n Mesh.id.name blend/factory_v279.blend
Mesh.id.name = "MECube"
$ ./target/release/blend_info -n Camera.lens blend/factory_v279.blend
Camera.lens = 35_f32
$ ./target/release/blend_info -n Object.obmat blend/factory_v279.blend
Object.obmat = [
0.68592066,
0.72767633,
0.0,
0.0,
-0.32401347,
0.30542085,
0.89539564,
0.0,
0.6515582,
-0.6141704,
0.4452714,
0.0,
7.4811316,
-6.50764,
5.343665,
1.0,
]
Object.obmat = [
1.0,
0.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
0.0,
1.0,
]
Object.obmat = [
-0.29086465,
0.95517117,
-0.05518906,
0.0,
-0.7711008,
-0.19988336,
0.60452473,
0.0,
0.5663932,
0.2183912,
0.79467225,
0.0,
4.0762453,
1.005454,
5.903862,
1.0,
]
Structs
Each Blender versions stores a number of DnaStrC blocks, which represent C structs in the original C code.
Each Blender C struct can have member entries, with a type and a name.
Functions
Each Blender C struct can have member entries and this function calculates a size for each of these entries.
Extract byte by index, if member type matches “char”, otherwise return 0_u8.
Extract 4 bytes in a row and interpret those as a “float”.
Try to extract an array of 2 floats.
Try to extract an array of 3 floats.
Try to extract an array of 4 floats.
Try to extract a String from a C struct with a “*name*” member.
Extract 4 bytes in a row and interpret those as a “int”.
Try to extract 16 “float” values and return them as an array of f32 values.
Try to extract 8 bytes in a row and interpret those as a pointer (in memory).
Extract 2 bytes in a row and interpret those as a “short”.
Try to extract an array of 3 shorts.
Print information about a pointer (byte position in binary .blend file vs. memory address).
Read a .blend
file to extract DNA information first.
Use the DNA types and structs returned by read_dna().