200 lines
7.7 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

##
## This file is part of the libsigrokdecode project.
##
## Copyright (C) 2011 Gareth McMullin <gareth@blacksphere.co.nz>
## Copyright (C) 2012-2014 Uwe Hermann <uwe@hermann-uwe.de>
## Copyright (C) 2022 DreamSourceLab <support@dreamsourcelab.com>
##
## This program 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 2 of the License, or
## (at your option) any later version.
##
## This program 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 this program; if not, see <http://www.gnu.org/licenses/>.
##
## 开发脚本的前提条件是必须掌握python语言
## 更复杂的协议请查考(0-i2c、0-spi、 0-uart、1-i2c、1-spi、1-uart)
## text block fill color table:
## [#EF2929,#F66A32,#FCAE3E,#FBCA47,#FCE94F,#CDF040,#8AE234,#4EDC44,#55D795,#64D1D2
## ,#729FCF,#D476C4,#9D79B9,#AD7FA8,#C2629B,#D7476F]
# 导出核心模块类c代码实现的类
import sigrokdecode as srd
# 协议模块类
class Decoder(srd.Decoder):
# 这里定义类的一些全局变量,有一些是底层框架要求必须要写的,其它提根据需要
# 自己加注意缩进不清楚请查看python手册
# 说明需要安装的python版本
api_version = 3
# 协议标识,必须唯一,这里我们用"example"给协议命名
id = 'example'
# 协议名称, 不一定要求跟标识一致
name = 'Example'
# 协议长名称
longname = 'example-lala'
# 简介内容
desc = 'This is an example of protocol'
# 开源协议
license = 'gplv2+'
# 接收的输入的数据源名,如果是多层协议一起工作,可使用上一个协议的输出名
inputs = ['logic']
# 输出的数据源名,多层协议模式下,可作为下层协议的输入数据源名
outputs = ['test']
# 适用范围标签
tags = ['Embedded/industrial']
# 必须要绑定的通道定义,将在界面上可见
# id:通道标识, 任意命名
# type:类型,根据需要设置一个值, -1:COMMON,0:SCLK,1:SDATA,2:ADATA
# name:标签名
# desc:该通道的说明
# 注意元组的最后的逗号不能少
channels = (
{'id': 'c1', type:0, 'name': 'c1', 'desc': 'chan1-input'},
)
# 可选通道,其它跟上面的一样
optional_channels = (
{'id': 'c2', type:0, 'name': 'c2', 'desc': 'chan2-input'},
)
# 提供给用户通过界面设置的参数,根据业务需要来定义
# 通过self.options[id]取值id就是各个项的id值比如下面的"wordsize"
options = (
# 这种参数是一个下拉列表
{'id': 'debug_bits', 'desc': 'Print each bit', 'default': 'no',
'values': ('yes', 'no')},
# 这是一个输入框
{'id': 'wordsize', 'desc': 'Data wordsize (# bus cycles)', 'default': 0},
)
# 解析结果项定义
# annotations里的每一项可以有2到3个属性当有个属性时第一个表示类型
# 类型对应0-16个颜色当类型范围在200-299时将绘制边沿箭头
annotations = (
('1', 'data1', 'test data1'),
('2', 'data2', 'test data2'),
('222', 'data3', 'test data3'),
)
# 解析结果行定义
annotation_rows = (
# (0,)表示可输出第1个定义的annotations类型
('lab1', 'row1', (0,)),
# (1,2)表示可输出第1个和第2个定义的annotations类型
('lab2', 'row2', (1,2)),
)
# 构造函数,自动被调用
def __init__(self):
# 这里调用一个类成员函数,完成一些参数的初始化
self.reset()
# 重置函数,在这里做一些重置和定义类私有变量工作
def reset(self):
# 定义一个私有变量count
self.count = 0
# 开始执行解码任务时由c底层代码自动调用一次
# 这里完成一些解码结果项annotation类型的注册
# 类型有: OUTPUT_ANNOUTPUT_PYTHONOUTPUT_BINARYOUTPUT_META
# self.register函数是c底层类提供的
def start(self):
self.out_ann = self.register(srd.OUTPUT_ANN)
# 定义一个输出函数
# a,b为采样位置的起点和终点
# ann为annotations定义的项序号
# data是一个列表列表里有个字符串它们将显示到屏幕
# annotation输出到哪一行由annotation_rows决定
# self.out_ann就是上面注册的消息类型了
# self.put是c底层类提供的函数
def put_ann(self, a, b, ann, data):
self.put(a, b, self.out_ann, [ann, data])
# 解码函数解码任务开始时由c底层代码调用
# 这里不断循环等待所有采样数据被处理完成
# 下面的示例代码是解析某一通道的数据,从向上边沿开始到向下边沿结束,输出它们的样品位置差值,
# 奇数次显示第二行偶数次显示在第一行我们只指定annotations里定义的序号
# 软件会自动根据annotation_rows的设置决定显示在哪一行
def decode(self):
# 次数计数
times = 0
# 向上边沿样品位置
rising_sample = 0
# 条件参数列表
flag_arr = [{0:'r'}, {0:'f'}]
# wait参数的条件序号
flag_dex = 0
# 不断循环,处理完所有数据
while True:
# 边沿条件
edge = flag_arr[flag_dex]
# 取一个条件调用wait函数找到符合条件的数据
# (a,b)是一个元组里边的变量数一定要跟channel数一致
(a,b) = self.wait(edge)
# 条件在向上边沿和向下边沿中切换
# 初始是向上边沿,取到样品位置后,切换成下向边沿
if flag_dex == 0:
flag_dex = 1
# 保存向上边沿采样位置
rising_sample = self.samplenum
else:
flag_dex = 0
# 打印次数计数加1
times += 1
# 向下边沿采样位置
falling_sample = self.samplenum
# 向下边沿和向上边沿采样位置差
v = falling_sample - rising_sample
# 转换成16进制的字符串
s = '%02X' % v
# 对times求余得值在0和1中变换,对应annotation的序号
ann = times % 2
self.put_ann(rising_sample, falling_sample, ann, [s])
# self.wait()可带参数,也可以不带参数,不带参数时将返回每个采样数据
# 参数{0:'r'} 0表示匹配channels第项绑定的通道'r'表示查找向上边沿
# wait函数可传多个条件与条件:{0:'f',:'r'}, 或条件:[{0:'f'},{:'r'}]
# h:高电平l:低电平r:向上边沿f:向下边沿e:向上沿或向下沿, n:要么0,要么1
# wait函数前的变量(a,b)对应的数量由定义的channels里的通道数决定包括可选通道
# optional_channels 。例如channels和optional_channels共定义了个通道
# 则变成(a,b,c,d) = self.wait(),共四个变量
# 底层模块提供的属性:
# 1. self.samplenum 当前wait()调用匹配结束的采样点位置
# 2. self.matched 本次调用wait()后所有通道的匹配结果信息是一个uint64类型数值
# 表示到63个通道的匹配信息通过位运算来获取具体信息。