Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Port FlatBuffers to Rust #4898

Merged
merged 37 commits into from
Sep 3, 2018
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
37 commits
Select commit Hold shift + click to select a range
93f9162
Port FlatBuffers to Rust: generator/runtime/tests.
rw Aug 29, 2018
66c6440
add idl_gen_rust.cpp to BUILD
rw Aug 31, 2018
d608827
remove the duplicate function `fill` and add some inline attributes
rw Aug 31, 2018
596017e
add comment in docs that Rust might add a verifier in the future
rw Aug 31, 2018
38b264c
generate tests/monsterdata_rust_wire.mon during rust tests (and gitig…
rw Aug 31, 2018
80725be
add rust to flatbuffers.md docs
rw Aug 31, 2018
129dd2d
docs
rw Aug 31, 2018
a865717
simplify nested asserts
rw Aug 31, 2018
2ed43e0
refactor vtable writing
rw Aug 31, 2018
9499632
remove duplicate assert
rw Aug 31, 2018
5138f85
comment
rw Aug 31, 2018
71a5ea4
use size_of for constants
rw Aug 31, 2018
9ea510c
comment
rw Aug 31, 2018
9736b9f
use auto instead of std::string in many places in the rust generator
rw Aug 31, 2018
3e74948
update generate_code.bat with rust
rw Aug 31, 2018
52d43ca
comments on reserved keywords; indentation of union accessors
rw Aug 31, 2018
b5b4631
delete test output
rw Aug 31, 2018
b5e5096
comment to explain why we do not inline a struct creation
rw Aug 31, 2018
77fb225
refactor some tests with macros
rw Aug 31, 2018
47ebb61
add support column for rust
rw Aug 31, 2018
ee51cc0
ergonomics/lifetimes tweaks in builder
rw Sep 1, 2018
a78edd0
no more need for ZeroTerminatedByteSlice
rw Sep 1, 2018
67004f3
wip for making Push::size a static method
rw Sep 1, 2018
894fe3e
simplifying push some more
rw Sep 1, 2018
fb28cad
delete old test
rw Sep 1, 2018
bd33f75
inlining, push
rw Sep 1, 2018
a3ef1de
Push::size is static
rw Sep 1, 2018
87cef86
start_vector requires push
rw Sep 2, 2018
bd2a5ed
more push/vector typing
rw Sep 2, 2018
36de894
no more phantomdata
rw Sep 2, 2018
3c03f30
good sample and docs
rw Sep 2, 2018
0de4628
add a todo to rust lib
rw Sep 2, 2018
7972b79
regenerate rust test code
rw Sep 2, 2018
f070f53
more alignment/size tests
rw Sep 2, 2018
8081c5a
remove dead code
rw Sep 2, 2018
acf48d0
doc tweak
rw Sep 2, 2018
4d005c1
revert old typescript file to master
rw Sep 2, 2018
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
simplifying push some more
  • Loading branch information
rw committed Sep 1, 2018
commit 894fe3e695ca3b097f2e2d538bb4dbbc9c814a25
26 changes: 18 additions & 8 deletions rust/flatbuffers/src/builder.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,11 @@

extern crate smallvec;

use std::borrow::Borrow;
use std::cmp::max;
use std::marker::PhantomData;
use std::mem::size_of;
use std::ptr::write_bytes;
use std::slice::from_raw_parts;

use endian_scalar::{read_scalar, emplace_scalar};
use primitives::*;
Expand Down Expand Up @@ -247,9 +247,20 @@ impl<'fbb> FlatBufferBuilder<'fbb> {
/// always safe, on any platform: bool, u8, i8, and any
/// FlatBuffers-generated struct.
#[inline]
pub fn create_vector_direct<'a: 'b, 'b, T: SafeSliceAccess + Push + Sized + 'b>(&'a mut self, data: &'b [T]) -> WIPOffset<Vector<'fbb, T>> {
pub fn create_vector_direct<'a: 'b, 'b, T: SafeSliceAccess + Push + Sized + 'b>(&'a mut self, items: &'b [T]) -> WIPOffset<Vector<'fbb, T>> {
self.assert_not_nested("create_vector_direct can not be called when a table or vector is under construction");
self.push(data);
let elem_size = size_of::<T>();
self.align(SIZE_UOFFSET + items.len() * elem_size, max(elem_size, SIZE_UOFFSET));
self.make_space(SIZE_UOFFSET + items.len() * elem_size);

let bytes = unsafe {
let ptr = items.as_ptr() as *const T as *const u8;
from_raw_parts(ptr, items.len() * elem_size)
};

self.owned_buf[self.head..self.head+bytes.len()].copy_from_slice(bytes);
self.push(items.len() as UOffsetT);

WIPOffset::new(self.used_space() as UOffsetT)
}

Expand All @@ -276,14 +287,13 @@ impl<'fbb> FlatBufferBuilder<'fbb> {
/// Speed-sensitive users may wish to reduce memory usage by creating the
/// vector manually: use `create_vector`, `push`, and `end_vector`.
#[inline]
pub fn create_vector<'a: 'b, 'b, T: Push + Copy + 'b, B: Borrow<T>>(&'a mut self, items: &'b [B]) -> WIPOffset<Vector<'fbb, T::Output>> {
pub fn create_vector<'a: 'b, 'b, T: Push + Copy + 'b>(&'a mut self, items: &'b [T]) -> WIPOffset<Vector<'fbb, T::Output>> {
let elem_size = size_of::<T>();
self.start_vector(elem_size, items.len());
// TODO(rw): precompute the space needed and call `make_space` only once
self.align(items.len() * elem_size, max(SIZE_SOFFSET, elem_size));
for i in (0..items.len()).rev() {
self.push(*items[i].borrow());
self.push(items[i]);
}
WIPOffset::new(self.end_vector::<T::Output>(items.len()).value())
WIPOffset::new(self.push::<UOffsetT>(items.len() as UOffsetT).value())
}

/// Get the byte slice for the data that has been written, regardless of
Expand Down
8 changes: 4 additions & 4 deletions rust/flatbuffers/src/primitives.rs
Original file line number Diff line number Diff line change
Expand Up @@ -112,12 +112,12 @@ impl<'a, T: 'a> WIPOffset<T> {

/// Return a wrapped value that brings its meaning as a union WIPOffset
/// into the type system.
#[inline]
#[inline(always)]
pub fn as_union_value(&self) -> WIPOffset<UnionWIPOffset> {
WIPOffset::new(self.0)
}
/// Get the underlying value.
#[inline]
#[inline(always)]
pub fn value(&self) -> UOffsetT {
self.0
}
Expand All @@ -126,7 +126,7 @@ impl<'a, T: 'a> WIPOffset<T> {
impl<T> Push for WIPOffset<T> {
type Output = ForwardsUOffset<T>;

#[inline]
#[inline(always)]
fn push(&self, dst: &mut [u8], rest: &[u8]) {
let n = (SIZE_UOFFSET + rest.len() - self.value() as usize) as UOffsetT;
emplace_scalar::<UOffsetT>(dst, n);
Expand All @@ -136,7 +136,7 @@ impl<T> Push for WIPOffset<T> {
impl<T> Push for ForwardsUOffset<T> {
type Output = Self;

#[inline]
#[inline(always)]
fn push(&self, dst: &mut [u8], rest: &[u8]) {
self.value().push(dst, rest);
}
Expand Down
32 changes: 0 additions & 32 deletions rust/flatbuffers/src/push.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,13 +14,9 @@
* limitations under the License.
*/

use std::cmp::max;
use std::mem::size_of;
use std::slice::from_raw_parts;

use endian_scalar::emplace_scalar;
use primitives::*;
use vector::{SafeSliceAccess, Vector};

/// Trait to abstract over functionality needed to write values. Used in
/// FlatBufferBuilder and implemented for generated types.
Expand All @@ -39,34 +35,6 @@ pub trait Push: Sized {
}
}

/// Push-able wrapper for slices of types that implement SafeSliceAccess.
impl<'a, T: SafeSliceAccess + Sized> Push for &'a [T] {
type Output = Vector<'a, u8>;

#[inline]
fn push(&self, dst: &mut [u8], _rest: &[u8]) {
let elem_sz = size_of::<T>();
let data = {
let ptr = self.as_ptr() as *const T as *const u8;
unsafe {
from_raw_parts(ptr, self.len() * elem_sz)
}
};
emplace_scalar::<UOffsetT>(&mut dst[..SIZE_UOFFSET], self.len() as UOffsetT);
dst[SIZE_UOFFSET..SIZE_UOFFSET+data.len()].copy_from_slice(data);
}

#[inline]
fn size(&self) -> usize {
SIZE_UOFFSET + self.len() * size_of::<T>()
}

#[inline]
fn alignment(&self) -> usize {
max(SIZE_UOFFSET, size_of::<T>())
}
}

/// Macro to implement Push for EndianScalar types.
macro_rules! impl_push_for_endian_scalar {
($ty:ident) => (
Expand Down