2016-04-13 18:03:57 +02:00
|
|
|
// Copyright 2015, 2016 Ethcore (UK) Ltd.
|
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
//! Parity interprocess hypervisor module
|
|
|
|
|
2016-04-14 19:56:06 +02:00
|
|
|
// while not in binary
|
|
|
|
#![allow(dead_code)]
|
|
|
|
|
2016-04-13 18:03:57 +02:00
|
|
|
pub mod service;
|
|
|
|
|
|
|
|
pub const HYPERVISOR_IPC_URL: &'static str = "ipc:///tmp/parity-internal-hyper-status.ipc";
|
|
|
|
|
|
|
|
use nanoipc;
|
2016-04-14 17:22:31 +02:00
|
|
|
use std::sync::{Arc,RwLock};
|
|
|
|
use hypervisor::service::*;
|
2016-04-13 18:03:57 +02:00
|
|
|
|
|
|
|
pub struct Hypervisor {
|
2016-04-14 17:22:31 +02:00
|
|
|
service: Arc<HypervisorService>,
|
2016-04-13 18:03:57 +02:00
|
|
|
ipc_worker: RwLock<nanoipc::Worker<HypervisorService>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Hypervisor {
|
|
|
|
/// initializes the Hypervisor service with the open ipc socket for incoming clients
|
|
|
|
pub fn init() -> Arc<Hypervisor>{
|
2016-04-14 17:22:31 +02:00
|
|
|
Hypervisor::with_url(HYPERVISOR_IPC_URL)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn with_url(addr: &str) -> Arc<Hypervisor>{
|
2016-04-14 17:49:25 +02:00
|
|
|
Hypervisor::with_url_and_service(addr, HypervisorService::new())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn with_url_and_service(addr: &str, service: Arc<HypervisorService>) -> Arc<Hypervisor> {
|
2016-04-14 17:22:31 +02:00
|
|
|
let mut worker = nanoipc::Worker::new(&service);
|
2016-04-14 19:56:06 +02:00
|
|
|
worker.add_reqrep(addr).expect("Hypervisor ipc worker can not start - critical!");
|
2016-04-13 18:03:57 +02:00
|
|
|
|
|
|
|
Arc::new(Hypervisor{
|
2016-04-14 17:22:31 +02:00
|
|
|
service: service,
|
|
|
|
ipc_worker: RwLock::new(worker),
|
2016-04-13 18:03:57 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2016-04-14 17:22:31 +02:00
|
|
|
pub fn modules_ready(&self) -> bool {
|
|
|
|
self.service.unchecked_count() == 0
|
|
|
|
}
|
|
|
|
|
2016-04-13 18:03:57 +02:00
|
|
|
/// Waits for every required module to check in
|
2016-04-14 17:22:31 +02:00
|
|
|
pub fn wait_for_startup(&self) {
|
|
|
|
let mut worker = self.ipc_worker.write().unwrap();
|
|
|
|
while !self.modules_ready() {
|
|
|
|
worker.poll()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
2016-04-14 17:49:25 +02:00
|
|
|
use std::sync::atomic::{AtomicBool,Ordering};
|
|
|
|
use std::sync::Arc;
|
|
|
|
use super::service::*;
|
|
|
|
use nanoipc;
|
2016-04-14 17:22:31 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_init() {
|
2016-04-14 19:43:14 +02:00
|
|
|
let url = "ipc:///tmp/test-parity-hypervisor-10.ipc";
|
2016-04-14 19:56:06 +02:00
|
|
|
let test_module_id = 8080u64;
|
2016-04-14 17:49:25 +02:00
|
|
|
|
2016-04-14 19:56:06 +02:00
|
|
|
let hypervisor = Hypervisor::with_url_and_service(url, HypervisorService::with_modules(vec![test_module_id]));
|
2016-04-14 17:22:31 +02:00
|
|
|
assert_eq!(false, hypervisor.modules_ready());
|
|
|
|
}
|
2016-04-13 18:03:57 +02:00
|
|
|
|
2016-04-14 17:22:31 +02:00
|
|
|
#[test]
|
|
|
|
fn can_wait_for_startup() {
|
2016-04-14 19:43:14 +02:00
|
|
|
let url = "ipc:///tmp/test-parity-hypervisor-20.ipc";
|
2016-04-14 17:49:25 +02:00
|
|
|
let test_module_id = 8080u64;
|
|
|
|
|
|
|
|
let hypervisor_ready = Arc::new(AtomicBool::new(false));
|
|
|
|
let hypervisor_ready_local = hypervisor_ready.clone();
|
|
|
|
|
|
|
|
::std::thread::spawn(move || {
|
|
|
|
while !hypervisor_ready.load(Ordering::Relaxed) { }
|
|
|
|
|
|
|
|
let client = nanoipc::init_client::<HypervisorServiceClient<_>>(url).unwrap();
|
2016-04-14 19:43:14 +02:00
|
|
|
client.handshake().unwrap();
|
2016-04-14 17:49:25 +02:00
|
|
|
client.module_ready(test_module_id);
|
|
|
|
});
|
|
|
|
|
|
|
|
let hypervisor = Hypervisor::with_url_and_service(url, HypervisorService::with_modules(vec![test_module_id]));
|
|
|
|
hypervisor_ready_local.store(true, Ordering::Relaxed);
|
2016-04-14 17:22:31 +02:00
|
|
|
hypervisor.wait_for_startup();
|
2016-04-14 17:49:25 +02:00
|
|
|
|
|
|
|
assert_eq!(true, hypervisor.modules_ready());
|
2016-04-13 18:03:57 +02:00
|
|
|
}
|
|
|
|
}
|