WaveDrom可以根据特定指令生成时序/波形/寄存器/逻辑电路图(SVG/PNG),并且也可以在浏览器中运行。同时提供了在线编辑器和教程、教程2。
信号(signal
)
名称(name
)和波形(wave
)
绘制信号时,需要向WaveDrom传递键为signal
,值为WaveLanes
的数组,WaveLane
一般至少需要包含两个属性:名称(name
)和波形(wave
),例如(在官方在线编辑器中打开):
{ "signal" : [{ "name": "clk", "wave": "P..........." }] }
波形的名称将显示在波形左侧,波形由一个字符串指定,每个字符都代表了一个周期的波形,每个字符代表的意义如下:
符号 | 含义 | 样例 | 渲染结果 |
---|---|---|---|
. | 延续上一周期的状态 | { signal : [{ name: ".", wave: "p..." }]} | |
p | 在周期开始时为上升沿的时钟信号,上升沿无箭头 | { signal : [{ name: ".", wave: "p.pp" }]} | |
P | 在周期开始时为上升沿的时钟信号,上升沿有箭头 | { signal : [{ name: ".", wave: "P.PP" }]} | |
n | 在周期开始时为下降沿的时钟信号,下降沿无箭头 | { signal : [{ name: ".", wave: "n.nn" }]} | |
N | 在周期开始时为下降沿的时钟信号,下降沿有箭头 | { signal : [{ name: ".", wave: "N.NN" }]} | |
0 | 低电平,有过渡 | { signal : [{ name: ".", wave: "0.01" }]} | |
l | 低电平,无过渡 | { signal : [{ name: ".", wave: "l.lh" }]} | |
1 | 高电平,有过渡 | { signal : [{ name: ".", wave: "1.10" }]} | |
h | 高电平,无过渡 | { signal : [{ name: ".", wave: "h.hl" }]} | |
z | 高阻态 | { signal : [{ name: ".", wave: "z.zz" }]} | |
d | 下拉(弱0) | { signal : [{ name: ".", wave: "d.dd" }]} | |
u | 上拉(弱1) | { signal : [{ name: ".", wave: "u.uu" }]} | |
x | 未定义 | { signal : [{ name: ".", wave: "x.xx" }]} | |
2 /= | 值(颜色为2) | { signal : [{ name: "2", wave: "2.22" }]} | |
3 | 值(颜色为3) | { signal : [{ name: "3", wave: "3.33" }]} | |
4 /5 /6 | 值(颜色为4/5/6) | { signal : [{ name: "4/5/6", wave: "4.56" }]} | |
7 /8 /9 | 值(颜色为7/8/9) | { signal : [{ name: "7/8/9", wave: "7.89" }]} | |
| | 延长前一个周期并绘制间隙 | { signal : [{ name: "|", wave: "xx|x" }]} |
这两个属性也可以没有,都没有时将显示一个空行,但仍可以添加其他属性(例如节点和连线)。
值名称(data
)
WaveLane
的data
属性指定按波形中值的名称,对应显示在wave
属性为2=3456789
及其延续周期的位置,值的顺序和data属性中字符串的顺序一致:
{ "signal" : [{ "name": "bus", "wave": "x.==.3x", "data": ["head", "", "tail"] }]}
分组
通过数组的第一项的值控制分组,可嵌套:
{ signal: [
['Slave',
['ctrl',
{name: 'ack', wave: 'x01x0.1x'},
],
{ name: 'rdata', wave: 'x.....4x', data: 'Q2'},
]
]}
周期(period
)和相位(phase
)
通过WaveLane
的period
和phase
项控制周期和相位,period
只能为正整数(也就是只能放大,可以通过设置主题缩短周期的间距):
{ signal: [
{ name: "CK", wave: "P.......", period: 2 },
{ name: "CMD", wave: "x.3x=x4x=x=x=x=x", data: "RAS NOP CAS NOP NOP NOP NOP", phase: 0.5 },
{ name: "DQ", wave: "z.........5555z.", data: "D0 D1 D2 D3" }
]}
节点(node
)
node
可以指定节点,小写字母会在周期起始处显示,大写字母不显示(一般用于连线)。相同连续的字符只会在最后一个字符对应的周期显示,.
代表空白:
{ signal: [
{ name: 'A', wave: '01........0....', node: '.a...c.D..j' },
],
}
连线(edge
)
通过指定连线(edge
)连接两个节点,并且可以在连线内部显示字符。每个连线由两个部分组成,第一部分为连线类型,第二部分为显示在连线上的字符,例如a~b t
代表平滑连线,并在连线上显示t
。连线有两种类别,一种是平滑的,一种是直线的:
连线类型 | 含义 | 样例 | 渲染结果 |
---|---|---|---|
~ | 平滑的连线 | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a~b']} | |
-~ | 上凸的平滑连线 | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a-~b']} | |
~- | 下凹的平滑连线 | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a~-b']} | |
- | 直接连接的直线连线 | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a-b']} | |
-|- | 中间转折的直线连线 | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a-|-b']} | |
-| | 上凸的直线连线 | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a-|b']} | |
|- | 下凹的直线连线 | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a|-b']} | |
+ | 在两端节点不显示(即节点为大写字母时)时显示两端带竖线的直线连线,在显示节点名称时行为与- 类似。 | { signal: [{ name: 'A', wave: '01.' }, { node: '.AB' },], edge: ['A+B']} | |
~> | 带单箭头的平滑连线,同时支持其他连线类型(例如-~> )和右箭头(例如<~ ) | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a~>b']} | |
<~> | 带双箭头的平滑连线,同时支持其他连线类型(例如<-~> ) | { signal: [{ name: 'A', wave: '01.', node: '.a.' }, { name: 'B', wave: '0.1', node: '..b' },], edge: ['a<~>b']} |
平滑连线样例:
{ signal: [
{ name: 'A', wave: '01........0....', node: '.a........j' },
{ name: 'B', wave: '0.1.......0.1..', node: '..b.......i' },
{ name: 'C', wave: '0..1....0...1..', node: '...c....h..' },
{ name: 'D', wave: '0...1..0.....1.', node: '....d..g...' },
{ name: 'E', wave: '0....10.......1', node: '.....ef....' }
],
edge: [
'a~b t1', 'c-~a t2', 'c-~>d time 3', 'd~-e',
'e~>f', 'f->g', 'g-~>h', 'h~>i some text', 'h~->j'
]
}
直线连线样例:
{ signal: [
{ name: 'A', wave: '01..0..', node: '.a..e..' },
{ name: 'B', wave: '0.1..0.', node: '..b..d.', phase:0.5 },
{ name: 'C', wave: '0..1..0', node: '...c..f' },
{ node: '...g..h' },
{ node: '...I..J', phase:0.5 },
{ name: 'D', wave: '0..1..0', phase:0.5 }
],
edge: [
'b-|a t1', 'a-|c t2', 'b-|-c t3', 'c-|->e t4', 'e-|>f more text',
'e|->d t6', 'c-g', 'f-h', 'g<->h 3 ms', 'I+J 5 ms'
]
}
适用于波形的其他配置
配置(config
)
水平缩放(hscale
)
通过指定配置(config
)的水平缩放(hscale
)值来控制水平缩放,该值必须为大于零的整数(也就是说只能放大),若要缩小周期,需要配置皮肤。
{ signal: [
{ name: "clk", wave: "p...." },],
config: { hscale: 2 }
}
皮肤(skin
)
通过指定配置(config
)的皮肤(skin
)添加不同的配色或是周期大小,可选的皮肤可以在这里查看,例如使用narrow
可以减小周期长度,dark
为深色皮肤。
{ signal: [
{ name: "clk", wave: "p...." },],
config: { skin: 'narrow' }
}
{ signal: [
{ name: "clk", wave: "p...." },],
config: { skin: 'dark' }
}
隔断(gap
)
在周期中央绘制隔断,类型有(
左隔断、)
右隔断、|
竖隔断、.
无隔断。
{ signal: [
{name: 'clock', wave: 'p.....'},
{name: 'valid', wave: '0101.0'},
{name: 'data', wave: 'x3x4.x', data: 'd0 d0'},
{},
{name: 'ready', wave: 'x1x01x'}
],
gaps: '( . | . . )' }
项眉(head
)/项脚(foot
)
配置(config
)的项眉(head
)/项脚(foot
)可以配置位于波形图上方和下方的文字(text
)和坐标(tick
对齐周期起始位置,tock
在周期中居中,其值为起始值,every
指定显示周期):
{signal: [
{name:'clk', wave: 'p....' },
{name:'Data', wave: 'x345x', data: 'a b c' },
{name:'Request', wave: '01..0' }
],
head:{
text:'WaveDrom example',
tick:0,
every:2
},
foot:{
text:'Figure 100',
tock:9
},
}
通过控制文本的tspan
属性来指定文本的样式,有预定义的类:h1
、h2
、h3
、h4
、h5
、h6
(控制字体大小),muted
、warning
、error
、info
、success
控制文本样式(颜色、斜体、粗细),其他tspan
属性也可以直接使用:
{signal: [
{name:'clk', wave: 'p.....PPPPp....' },
{name:'dat', wave: 'x....2345x.....', data: 'a b c d' },
{name:'req', wave: '0....1...0.....' }
],
head: {text:
['tspan',
['tspan', {class:'error h1'}, 'error '],
['tspan', {class:'warning h2'}, 'warning '],
['tspan', {class:'info h3'}, 'info '],
['tspan', {class:'success h4'}, 'success '],
['tspan', {class:'muted h5'}, 'muted '],
['tspan', {class:'h6'}, 'h6 '],
'default ',
['tspan', {fill:'pink', 'font-weight':'bold', 'font-style':'italic'}, 'pink-bold-italic']
]
},
foot: {text:
['tspan', 'E=mc',
['tspan', {dy:'-5'}, '2'],
['tspan', {dy: '5'}, '. '],
['tspan', {'font-size':'25'}, 'B '],
['tspan', {'text-decoration':'overline'},'over '],
['tspan', {'text-decoration':'underline'},'under '],
['tspan', {'baseline-shift':'sub'}, 'sub '],
['tspan', {'baseline-shift':'super'}, 'super ']
],tock:-5
}
}
寄存器(reg
)
名称(name
)和比特数(bit
)
名称(name
)和比特数(bit
)是寄存器最基本的两个属性,名称将显示在寄存器框内且可以为空,比特数代表此与区域的长度。列表的顺序是从高位到低位,一个寄存器默认在一行中显示。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3},
{ bits: 7}
]}
类型(type
)
类型(type
)指定寄存器的颜色:
{reg:[
{ bits: 1, name:""},
{ bits: 1, type: 0},
{ bits: 1, type: 1},
{ bits: 1, type: 2},
{ bits: 1, type: 3},
{ bits: 1, type: 4},
{ bits: 1, type: 5},
{ bits: 1, type: 6},
{ bits: 1, type: 7},
{ bits: 1, type: 8},
{ bits: 1, type: 9},
]}
属性(attr
)
属性(attr
)指定寄存器下方文字或二进制数字并按列表顺序依次显示,当属性为文字时直接显示,为数字时将展开为二进制,若数字的二进制超过最大范围则截取有效部分:
{reg: [
{bits: 7, name: 'opcode', attr: 'OP'},
{bits: 5, name: 'rd', attr: 'dest'},
{bits: 3, name: 'func3', attr: 'T0 T1 T2 T3 T4 T5 T6 T7 T8 T9'.split(' '), type: 4},
{bits: 5, name: 'rs1', attr: 'src1'},
{bits: 5, name: 'rs2', attr: 'src2'},
{bits: 7, name: 'funct7', attr: [0, 1, 2, 3, 4, 5, 8, 16, 32, 64]}
]}
旋转(rotate
)
指定寄存器名称的旋转角度,顺时针旋转为正。
{reg:[
{name: 'label -90', bits: 1, rotate: -90},
{name: 'label -45', bits: 1, rotate: -45},
{name: 'label 45', bits: 1, rotate: 45},
{name: 'label 90', bits: 1, rotate: 90},
{name: 'label 0', bits: 1, rotate: 0},
],config:{vspace:100}}
宽度(hspace
)、高度(vspace
)
指定整个寄存器的宽度和高度。
{reg:[
{name: 'IPO', bits: 8, attr: 'RO'},
{ bits: 7},
{name: 'BRK', bits: 5, attr: ['RW', 'FOO'], type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{hspace: 1200, vspace: 150}}
多行(lanes
)
指定以多少行显示寄存器,每行的寄存器长度一致。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{lanes:2}}
行号(label
)
指定寄存器左侧(left
)或者右侧(right
)的行号,参数为起始值。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{lanes:2, label: {right: 2}}}
位数(bit
)
当位数大于已有的寄存器长度时,剩余长度用空白填充。当位数小于已有的寄存器长度时,多余的寄存器将不会显示。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{bits: 48}}
水平顺序(vflip
)/垂直顺序(hflip
)
默认顺序为水平方向上高位在低位在后,垂直方向上为高位在上低位在下,通过水平顺序(vflip
)/垂直顺序(hflip
)可以改变该顺序。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{vflip: true}}
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{lanes:2, hflip: true}}
紧凑(compact
)
行之间无空隙。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{lanes:2, compact: true}}
字号(fontsize
)/间隙(margin
)
通过fontsize
指定字号。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{fontsize: 20}}
通过margin
指定与边缘的距离,包括上(top
)下(bottom
)左(left
)右(right
)四个方向。
{reg:[
{name: 'IPO', bits: 8},
{ bits: 7},
{name: 'BRK', bits: 5, type: 4},
{name: 'CPK', bits: 2},
{name: 'Clear', bits: 3, type: 5},
{ bits: 7}
],config:{margin: {left: 60}}}
逻辑电路(assign
)
逻辑电路的每个单元由输出和输入构成,输出可为一个字符串或门符号,输入为一个或多个字符串或逻辑电路单元。
门类型 | 含义 | 样例 | 渲染结果 |
---|---|---|---|
& | 与门 | { assign:[["&","a","b"]]} | |
~& | 与非门 | { assign:[["~&","a","b"]]} | |
| | 或门 | { assign:[["|","a","b"]]} | |
~| | 或非门 | { assign:[["~|","a","b"]]} | |
~ | 非门 | { assign:[["~","a"]]} | |
^ | 异或门 | { assign:[["^","a","b"]]} | |
~^ | 同或门 | { assign:[["~^","a","b"]]} | |
AND | (IEC 60617)与门 | { assign:[["AND","a","b"]]} | |
NAND | (IEC 60617)与非门 | { assign:[["NAND","a","b"]]} | |
OR | (IEC 60617)或门 | { assign:[["OR","a","b"]]} | |
NOR | (IEC 60617)或非门 | { assign:[["NOR","a","b"]]} | |
INV | (IEC 60617)非门 | { assign:[["INV","a"]]} | |
XOR | (IEC 60617)异或门 | { assign:[["XOR","a","b"]]} | |
XNOR | (IEC 60617)同或门 | { assign:[["XNOR","a","b"]]} | |
BUF | (IEC 60617)缓冲门 | { assign:[["BUF","a"]]} |
输出为字符串或门的样例:
{ assign:[["gate","a"],
["gate", "a", "b"]]}
输入既可以是字符串,也可以是逻辑电路单元,因此逻辑电路是可嵌套的:
{ assign:[
["out",
["|",
["&", ["~", "a"], "b"],
["&", ["~", "b"], "a"]
]
]
]}