/* * Copyright 2020 Fluence Labs Limited * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #![allow(improper_ctypes)] use fluence::fce; use fluence::module_manifest; use fluence::WasmLoggerBuilder; use fluence::MountedBinaryResult; module_manifest!(); /// Log level can be changed by `RUST_LOG` env as well. pub fn main() { WasmLoggerBuilder::new().build().unwrap(); } #[fce] pub fn download(url: String) -> String { log::info!("download called with url {}\n", url); unsafe { log::info!( "download track: {} {} {} {} {}", *ALLOCS.get_mut(), *DEALLOCS.get_mut(), *REALLOCSA.get_mut(), *REALLOCSD.get_mut(), *ALLOCS.get_mut() - *DEALLOCS.get_mut() - *REALLOCSD.get_mut() + *REALLOCSA.get_mut() ); } let result = curl(vec![url]); let res = String::from_utf8(result.stdout).unwrap(); log::info!("download ended with {}\n", res); res } /// Permissions in `Config.toml` should exist to use host functions. #[fce] #[link(wasm_import_module = "host")] extern "C" { fn curl(cmd: Vec) -> MountedBinaryResult; } use std::alloc::{GlobalAlloc, System, Layout}; #[global_allocator] static GLOBAL_ALLOCATOR: WasmTracingAllocator = WasmTracingAllocator(System); #[derive(Debug)] pub struct WasmTracingAllocator(pub A) where A: GlobalAlloc; use std::sync::atomic::AtomicUsize; static mut ALLOCS: AtomicUsize = AtomicUsize::new(0); static mut DEALLOCS: AtomicUsize = AtomicUsize::new(0); static mut REALLOCSA: AtomicUsize = AtomicUsize::new(0); static mut REALLOCSD: AtomicUsize = AtomicUsize::new(0); unsafe impl GlobalAlloc for WasmTracingAllocator where A: GlobalAlloc, { unsafe fn alloc(&self, layout: Layout) -> *mut u8 { let size = layout.size(); let t = *ALLOCS.get_mut(); *ALLOCS.get_mut() = t + size; let pointer = self.0.alloc(layout); pointer } unsafe fn dealloc(&self, pointer: *mut u8, layout: Layout) { let size = layout.size(); let t = *DEALLOCS.get_mut(); *DEALLOCS.get_mut() = t + size; self.0.dealloc(pointer, layout); } unsafe fn alloc_zeroed(&self, layout: Layout) -> *mut u8 { let size = layout.size(); let t = *ALLOCS.get_mut(); *ALLOCS.get_mut() = t + size; let pointer = self.0.alloc_zeroed(layout); pointer } unsafe fn realloc(&self, old_pointer: *mut u8, layout: Layout, new_size: usize) -> *mut u8 { let old_size = layout.size(); let t = *REALLOCSD.get_mut(); *REALLOCSD.get_mut() = t + old_size; let t = *REALLOCSA.get_mut(); *REALLOCSA.get_mut() = t + layout.size(); let new_pointer = self.0.realloc(old_pointer, layout, new_size); new_pointer } }