Blame servo/components/layout/layout_debug.rs

Packit f0b94e
/* This Source Code Form is subject to the terms of the Mozilla Public
Packit f0b94e
 * License, v. 2.0. If a copy of the MPL was not distributed with this
Packit f0b94e
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
Packit f0b94e
Packit f0b94e
//! Supports writing a trace file created during each layout scope
Packit f0b94e
//! that can be viewed by an external tool to make layout debugging easier.
Packit f0b94e
Packit f0b94e
use flow::GetBaseFlow;
Packit f0b94e
use flow_ref::FlowRef;
Packit f0b94e
use serde_json::{to_string, to_value, Value};
Packit f0b94e
use std::borrow::ToOwned;
Packit f0b94e
use std::cell::RefCell;
Packit f0b94e
use std::fs::File;
Packit f0b94e
use std::io::Write;
Packit f0b94e
#[cfg(debug_assertions)]
Packit f0b94e
use std::sync::atomic::{ATOMIC_USIZE_INIT, AtomicUsize, Ordering};
Packit f0b94e
Packit f0b94e
thread_local!(static STATE_KEY: RefCell<Option<State>> = RefCell::new(None));
Packit f0b94e
Packit f0b94e
#[cfg(debug_assertions)]
Packit f0b94e
static DEBUG_ID_COUNTER: AtomicUsize = ATOMIC_USIZE_INIT;
Packit f0b94e
Packit f0b94e
pub struct Scope;
Packit f0b94e
Packit f0b94e
#[macro_export]
Packit f0b94e
macro_rules! layout_debug_scope(
Packit f0b94e
    ($($arg:tt)*) => (
Packit f0b94e
        if cfg!(debug_assertions) {
Packit f0b94e
            layout_debug::Scope::new(format!($($arg)*))
Packit f0b94e
        } else {
Packit f0b94e
            layout_debug::Scope
Packit f0b94e
        }
Packit f0b94e
    )
Packit f0b94e
);
Packit f0b94e
Packit f0b94e
#[derive(Serialize)]
Packit f0b94e
struct ScopeData {
Packit f0b94e
    name: String,
Packit f0b94e
    pre: Value,
Packit f0b94e
    post: Value,
Packit f0b94e
    children: Vec<Box<ScopeData>>,
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
impl ScopeData {
Packit f0b94e
    fn new(name: String, pre: Value) -> ScopeData {
Packit f0b94e
        ScopeData {
Packit f0b94e
            name: name,
Packit f0b94e
            pre: pre,
Packit f0b94e
            post: Value::Null,
Packit f0b94e
            children: vec!(),
Packit f0b94e
        }
Packit f0b94e
    }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
struct State {
Packit f0b94e
    flow_root: FlowRef,
Packit f0b94e
    scope_stack: Vec<Box<ScopeData>>,
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
/// A layout debugging scope. The entire state of the flow tree
Packit f0b94e
/// will be output at the beginning and end of this scope.
Packit f0b94e
impl Scope {
Packit f0b94e
    pub fn new(name: String) -> Scope {
Packit f0b94e
        STATE_KEY.with(|ref r| {
Packit f0b94e
            if let Some(ref mut state) = *r.borrow_mut() {
Packit f0b94e
                let flow_trace = to_value(&state.flow_root.base()).unwrap();
Packit f0b94e
                let data = Box::new(ScopeData::new(name.clone(), flow_trace));
Packit f0b94e
                state.scope_stack.push(data);
Packit f0b94e
            }
Packit f0b94e
        });
Packit f0b94e
        Scope
Packit f0b94e
    }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
#[cfg(debug_assertions)]
Packit f0b94e
impl Drop for Scope {
Packit f0b94e
    fn drop(&mut self) {
Packit f0b94e
        STATE_KEY.with(|ref r| {
Packit f0b94e
            if let Some(ref mut state) = *r.borrow_mut() {
Packit f0b94e
                let mut current_scope = state.scope_stack.pop().unwrap();
Packit f0b94e
                current_scope.post = to_value(&state.flow_root.base()).unwrap();
Packit f0b94e
                let previous_scope = state.scope_stack.last_mut().unwrap();
Packit f0b94e
                previous_scope.children.push(current_scope);
Packit f0b94e
            }
Packit f0b94e
        });
Packit f0b94e
    }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
/// Generate a unique ID. This is used for items such as Fragment
Packit f0b94e
/// which are often reallocated but represent essentially the
Packit f0b94e
/// same data.
Packit f0b94e
#[cfg(debug_assertions)]
Packit f0b94e
pub fn generate_unique_debug_id() -> u16 {
Packit f0b94e
    DEBUG_ID_COUNTER.fetch_add(1, Ordering::SeqCst) as u16
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
/// Begin a layout debug trace. If this has not been called,
Packit f0b94e
/// creating debug scopes has no effect.
Packit f0b94e
pub fn begin_trace(flow_root: FlowRef) {
Packit f0b94e
    assert!(STATE_KEY.with(|ref r| r.borrow().is_none()));
Packit f0b94e
Packit f0b94e
    STATE_KEY.with(|ref r| {
Packit f0b94e
        let flow_trace = to_value(&flow_root.base()).unwrap();
Packit f0b94e
        let state = State {
Packit f0b94e
            scope_stack: vec![Box::new(ScopeData::new("root".to_owned(), flow_trace))],
Packit f0b94e
            flow_root: flow_root.clone(),
Packit f0b94e
        };
Packit f0b94e
        *r.borrow_mut() = Some(state);
Packit f0b94e
    });
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
/// End the debug layout trace. This will write the layout
Packit f0b94e
/// trace to disk in the current directory. The output
Packit f0b94e
/// file can then be viewed with an external tool.
Packit f0b94e
pub fn end_trace(generation: u32) {
Packit f0b94e
    let mut thread_state = STATE_KEY.with(|ref r| r.borrow_mut().take().unwrap());
Packit f0b94e
    assert_eq!(thread_state.scope_stack.len(), 1);
Packit f0b94e
    let mut root_scope = thread_state.scope_stack.pop().unwrap();
Packit f0b94e
    root_scope.post = to_value(&thread_state.flow_root.base()).unwrap();
Packit f0b94e
Packit f0b94e
    let result = to_string(&root_scope).unwrap();
Packit f0b94e
    let mut file = File::create(format!("layout_trace-{}.json", generation)).unwrap();
Packit f0b94e
    file.write_all(result.as_bytes()).unwrap();
Packit f0b94e
}