Merge pull request #42 from NikVolf/data-segments

Data segment builder
This commit is contained in:
Nikolay Volf 2017-06-09 15:40:09 +03:00 committed by GitHub
commit 2db02817aa
4 changed files with 84 additions and 2 deletions

49
src/builder/data.rs Normal file
View File

@ -0,0 +1,49 @@
use super::invoke::{Identity, Invoke};
use elements;
pub struct DataSegmentBuilder<F=Identity> {
callback: F,
// todo: add mapper once multiple memory refs possible
mem_index: u32,
offset: elements::InitExpr,
value: Vec<u8>,
}
impl DataSegmentBuilder {
pub fn new() -> Self {
DataSegmentBuilder::with_callback(Identity)
}
}
impl<F> DataSegmentBuilder<F> {
pub fn with_callback(callback: F) -> Self {
DataSegmentBuilder {
callback: callback,
mem_index: 0,
offset: elements::InitExpr::empty(),
value: Vec::new(),
}
}
pub fn offset(mut self, opcode: elements::Opcode) -> Self {
self.offset = elements::InitExpr::new(vec![opcode, elements::Opcode::End]);
self
}
pub fn value(mut self, value: Vec<u8>) -> Self {
self.value = value;
self
}
}
impl<F> DataSegmentBuilder<F> where F: Invoke<elements::DataSegment> {
pub fn build(self) -> F::Result {
self.callback.invoke(
elements::DataSegment::new(
self.mem_index,
self.offset,
self.value,
)
)
}
}

View File

@ -56,7 +56,6 @@ impl<F> GlobalBuilder<F> where F: Invoke<elements::GlobalEntry> {
}
}
impl<F> Invoke<elements::ValueType> for GlobalBuilder<F> {
type Result = Self;
fn invoke(self, the_type: elements::ValueType) -> Self {

View File

@ -9,6 +9,7 @@ mod memory;
mod table;
mod export;
mod global;
mod data;
pub use self::module::{module, from_module, ModuleBuilder};
pub use self::code::{signatures, signature, function};

View File

@ -2,7 +2,7 @@ use super::invoke::{Invoke, Identity};
use super::code::{self, SignaturesBuilder, FunctionBuilder};
use super::memory::{self, MemoryBuilder};
use super::table::{self, TableBuilder};
use super::{import, export, global};
use super::{import, export, global, data};
use elements;
/// Module builder
@ -321,6 +321,17 @@ impl<F> ModuleBuilder<F> where F: Invoke<elements::Module> {
global::GlobalBuilder::with_callback(self)
}
/// Add data segment to the builder
pub fn with_data_segment(mut self, segment: elements::DataSegment) -> Self {
self.module.data.entries_mut().push(segment);
self
}
/// Data entry builder
pub fn data(self) -> data::DataSegmentBuilder<Self> {
data::DataSegmentBuilder::with_callback(self)
}
/// Build module (final step)
pub fn build(self) -> F::Result {
self.callback.invoke(self.module.into())
@ -414,6 +425,16 @@ impl<F> Invoke<elements::GlobalEntry> for ModuleBuilder<F>
}
}
impl<F> Invoke<elements::DataSegment> for ModuleBuilder<F>
where F: Invoke<elements::Module>
{
type Result = Self;
fn invoke(self, segment: elements::DataSegment) -> Self {
self.with_data_segment(segment)
}
}
/// Start new module builder
pub fn module() -> ModuleBuilder {
ModuleBuilder::new()
@ -466,4 +487,16 @@ mod tests {
assert_eq!(module.global_section().expect("global section to exist").entries().len(), 1);
}
#[test]
fn data() {
let module = module()
.data()
.offset(::elements::Opcode::I32Const(16))
.value(vec![0u8, 15, 10, 5, 25])
.build()
.build();
assert_eq!(module.data_section().expect("data section to exist").entries().len(), 1);
}
}