2013-09-18 22:18:38 -07:00
|
|
|
// Copyright 2013 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
//! Markdown formatting for rustdoc
|
|
|
|
//!
|
|
|
|
//! This module implements markdown formatting through the sundown C-library
|
|
|
|
//! (bundled into the rust runtime). This module self-contains the C bindings
|
|
|
|
//! and necessary legwork to render markdown, and exposes all of the
|
|
|
|
//! functionality through a unit-struct, `Markdown`, which has an implementation
|
|
|
|
//! of `fmt::Default`. Example usage:
|
|
|
|
//!
|
|
|
|
//! ```rust
|
|
|
|
//! let s = "My *markdown* _text_";
|
|
|
|
//! let html = format!("{}", Markdown(s));
|
|
|
|
//! // ... something using html
|
|
|
|
//! ```
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
use std::fmt;
|
2013-09-23 16:55:48 -07:00
|
|
|
use std::libc;
|
2013-11-10 22:46:32 -08:00
|
|
|
use std::io;
|
2013-09-23 16:55:48 -07:00
|
|
|
use std::vec;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// A unit struct which has the `fmt::Default` trait implemented. When
|
|
|
|
/// formatted, this struct will emit the HTML corresponding to the rendered
|
|
|
|
/// version of the contained markdown string.
|
2013-09-18 22:18:38 -07:00
|
|
|
pub struct Markdown<'self>(&'self str);
|
|
|
|
|
2013-09-23 16:55:48 -07:00
|
|
|
static OUTPUT_UNIT: libc::size_t = 64;
|
2013-09-23 17:52:57 -07:00
|
|
|
static MKDEXT_NO_INTRA_EMPHASIS: libc::c_uint = 1 << 0;
|
|
|
|
static MKDEXT_TABLES: libc::c_uint = 1 << 1;
|
|
|
|
static MKDEXT_FENCED_CODE: libc::c_uint = 1 << 2;
|
|
|
|
static MKDEXT_AUTOLINK: libc::c_uint = 1 << 3;
|
|
|
|
static MKDEXT_STRIKETHROUGH: libc::c_uint = 1 << 4;
|
|
|
|
static MKDEXT_SPACE_HEADERS: libc::c_uint = 1 << 6;
|
|
|
|
static MKDEXT_SUPERSCRIPT: libc::c_uint = 1 << 7;
|
|
|
|
static MKDEXT_LAX_SPACING: libc::c_uint = 1 << 8;
|
2013-09-23 16:55:48 -07:00
|
|
|
|
|
|
|
type sd_markdown = libc::c_void; // this is opaque to us
|
|
|
|
|
|
|
|
// this is a large struct of callbacks we don't use
|
|
|
|
type sd_callbacks = [libc::size_t, ..26];
|
|
|
|
|
|
|
|
struct html_toc_data {
|
|
|
|
header_count: libc::c_int,
|
|
|
|
current_level: libc::c_int,
|
|
|
|
level_offset: libc::c_int,
|
|
|
|
}
|
|
|
|
|
|
|
|
struct html_renderopt {
|
|
|
|
toc_data: html_toc_data,
|
|
|
|
flags: libc::c_uint,
|
|
|
|
link_attributes: Option<extern "C" fn(*buf, *buf, *libc::c_void)>,
|
|
|
|
}
|
|
|
|
|
|
|
|
struct buf {
|
|
|
|
data: *u8,
|
|
|
|
size: libc::size_t,
|
|
|
|
asize: libc::size_t,
|
|
|
|
unit: libc::size_t,
|
|
|
|
}
|
|
|
|
|
|
|
|
// sundown FFI
|
2013-10-30 23:22:04 -07:00
|
|
|
#[link_args = "-lsundown"]
|
2013-09-23 16:55:48 -07:00
|
|
|
extern {
|
|
|
|
fn sdhtml_renderer(callbacks: *sd_callbacks,
|
|
|
|
options_ptr: *html_renderopt,
|
|
|
|
render_flags: libc::c_uint);
|
|
|
|
fn sd_markdown_new(extensions: libc::c_uint,
|
|
|
|
max_nesting: libc::size_t,
|
|
|
|
callbacks: *sd_callbacks,
|
|
|
|
opaque: *libc::c_void) -> *sd_markdown;
|
|
|
|
fn sd_markdown_render(ob: *buf,
|
|
|
|
document: *u8,
|
|
|
|
doc_size: libc::size_t,
|
|
|
|
md: *sd_markdown);
|
|
|
|
fn sd_markdown_free(md: *sd_markdown);
|
|
|
|
|
|
|
|
fn bufnew(unit: libc::size_t) -> *buf;
|
|
|
|
fn bufrelease(b: *buf);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
fn render(w: &mut io::Writer, s: &str) {
|
|
|
|
// This code is all lifted from examples/sundown.c in the sundown repo
|
|
|
|
unsafe {
|
|
|
|
let ob = bufnew(OUTPUT_UNIT);
|
2013-09-23 17:52:57 -07:00
|
|
|
let extensions = MKDEXT_NO_INTRA_EMPHASIS | MKDEXT_TABLES |
|
|
|
|
MKDEXT_FENCED_CODE | MKDEXT_AUTOLINK |
|
|
|
|
MKDEXT_STRIKETHROUGH;
|
2013-09-23 16:55:48 -07:00
|
|
|
let options = html_renderopt {
|
|
|
|
toc_data: html_toc_data {
|
|
|
|
header_count: 0,
|
|
|
|
current_level: 0,
|
|
|
|
level_offset: 0,
|
|
|
|
},
|
|
|
|
flags: 0,
|
|
|
|
link_attributes: None,
|
|
|
|
};
|
|
|
|
let callbacks: sd_callbacks = [0, ..26];
|
|
|
|
|
|
|
|
sdhtml_renderer(&callbacks, &options, 0);
|
2013-09-23 17:52:57 -07:00
|
|
|
let markdown = sd_markdown_new(extensions, 16, &callbacks,
|
2013-09-23 16:55:48 -07:00
|
|
|
&options as *html_renderopt as *libc::c_void);
|
|
|
|
|
2013-11-21 15:42:55 -08:00
|
|
|
s.as_imm_buf(|data, len| {
|
2013-09-23 16:55:48 -07:00
|
|
|
sd_markdown_render(ob, data, len as libc::size_t, markdown);
|
2013-11-21 15:42:55 -08:00
|
|
|
});
|
2013-09-23 16:55:48 -07:00
|
|
|
sd_markdown_free(markdown);
|
|
|
|
|
2013-11-21 15:42:55 -08:00
|
|
|
vec::raw::buf_as_slice((*ob).data, (*ob).size as uint, |buf| {
|
2013-09-23 16:55:48 -07:00
|
|
|
w.write(buf);
|
2013-11-21 15:42:55 -08:00
|
|
|
});
|
2013-09-23 16:55:48 -07:00
|
|
|
|
|
|
|
bufrelease(ob);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
impl<'self> fmt::Default for Markdown<'self> {
|
|
|
|
fn fmt(md: &Markdown<'self>, fmt: &mut fmt::Formatter) {
|
2013-09-23 16:55:48 -07:00
|
|
|
// This is actually common enough to special-case
|
2013-09-18 22:18:38 -07:00
|
|
|
if md.len() == 0 { return; }
|
2013-09-23 16:55:48 -07:00
|
|
|
render(fmt.buf, md.as_slice());
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|