活动介绍

VHDL中泛型子程序的使用与实践

立即解锁
发布时间: 2025-08-16 01:28:52 阅读量: 2 订阅数: 11
PDF

VHDL设计指南:从入门到精通

### VHDL 中泛型子程序的使用与实践 #### 1. 抽象数据类型(ADT)使用风险 在复杂的行为模型中使用抽象数据类型(ADT)有很多优势,但也存在一定风险。VHDL 无法隐藏 ADT 底层数据结构的具体细节,因为类型声明必须写在包声明中。这意味着 ADT 用户可能会利用这些信息,不通过 ADT 过程和函数来修改数据结构。例如,若一个 ADT 操作只是更新一个记录元素,用户可能会直接更新记录以避免过程调用的开销。然而,现代编译器和计算机使这种“优化”变得不必要,而且用户可能会无意中破坏数据结构。因此,在 VHDL 中使用 ADT 时,用户需要避免这种诱惑,遵守 ADT 接口中的约定,一点自律在建模过程中会带来显著的好处。 #### 2. 子程序中的泛型列表 泛型列表在子程序中允许我们描述可用于各种类型操作数的操作,就像在包中允许描述可复用的容器数据类型一样。 ##### 2.1 带泛型列表的子程序语法 - **过程**: ```plaintext subprogram_body ⇐ procedure identifier [ generic ( generic_interface_list ) ] [ [ parameter ] ( parameter_interface_list ) ] is { subprogram_declarative_part } begin { sequential_statement } end [ procedure ] [ identifier ] ; ``` - **函数**: ```plaintext subprogram_body ⇐ [ pure I impure ] function identifier [ generic ( generic_interface_list ) ] [ [ parameter ] ( parameter_interface_list ) ] return type_mark is { subprogram_declarative_item } begin { sequential_statement } end [ function ] [ identifier ] ; ``` 带有泛型列表的子程序称为未实例化的子程序。可选关键字 `parameter` 可用于明确泛型列表和参数列表的分界,对于没有泛型的子程序通常省略该关键字,对于未实例化的子程序是否包含该关键字可根据个人喜好决定。 ##### 2.2 子程序声明与实例化 VHDL 允许将子程序声明分为两部分,一部分仅包含头部,另一部分包含头部和主体。在实例化未实例化的子程序时,需要使用子程序实例化声明,例如: - **过程实例化**: ```plaintext subprogram_instantiation_declaration ⇐ procedure identifier is new uninstantiated_subprogram_name [ signature ] [ generic map ( generic_association_list ) ] ; ``` - **函数实例化**: ```plaintext subprogram_instantiation_declaration ⇐ function identifier is new uninstantiated_subprogram_name [ signature ] [ generic map ( generic_association_list ) ] ; ``` 实例化后,使用实例名称调用实例。如果所有形式泛型都有默认值,可以省略 `generic map` 以使用默认值。 ##### 2.3 示例:通用交换过程 ```vhdl procedure swap generic ( type T ) parameter ( a, b : inout T ) is variable temp : T; begin temp := a; a := b; b := temp; end procedure swap; procedure int_swap is new swap generic map ( T => integer ); procedure vec_swap is new swap generic map ( T => bit_vector(0 to 7) ); variable a_int, b_int : integer; variable a_vec, b_vec : bit_vector(0 to 7); int_swap(a_int, b_int); vec_swap(a_vec, b_vec); ``` 需要注意,不能直接调用未实例化的 `swap` 过程,并且不能用无约束类型作为实际泛型类型来实例化 `swap` 过程,因为过程内部使用该类型声明了一个变量。 ##### 2.4 示例:建立时间检查过程 ```vhdl package timing_pkg is procedure check_setup generic ( type signal_type; type clk_type; clk_active_value : clk_type; T_su : delay_length ) ( signal s : signal_type; signal clk : clk_type ); ... end package timing_pkg; package body timing_pkg is procedure check_setup generic ( type signal_type; type clk_type; clk_active_value : clk_type; T_su : delay_length ) ( signal s : signal_type; signal clk : clk_type ) is begin if clk'event and clk = clk_active_value then assert s'last_event >= T_su report "Setup time violation" severity error; end if; end procedure check_setup; ... end package body timing_pkg; use work.timing_pkg.all; procedure check_normal_setup is new check_setup generic map ( signal_type => std_ulogic, clk_type => std_ulogic, clk_active_value => '1', T_su => 200ps ); procedure check_normal_setup is new check_setup generic map ( signal_type => std_ulogic_vector, clk_type => std_ulogic, clk_active_value => '1', T_su => 200ps ); procedure check_long_setup is new check_setup generic map ( signal_type => std_ulogic_vector, clk_type => std_ulogic, clk_active_value => '1', T_su => 300ps ); signal status : std_ulogic; signal data_in, result : std_ulogic_vector(23 downto 0); check_normal_setup(status, clk); check_normal_setup(result, clk); check_long_setup(data_in, clk); ``` 在这个例子中,时钟信号的活动值和建立时间间隔值被绑定到过程实例的定义中,不需要作为单独的参数提供。 ##### 2.5 签名的使用 VHDL 允许重载子程序,并根据调用中参数的类型使用参数和结果类型配置文件来区分它们。当需要在调用之外命名子程序时,可以写一个签名来表明我们指的是哪个重载版本。例如,在别名声明中就可以使用签名。签名列出参数类型,对于函数还列出返回类型,都用方括号括起来。如果一个未实例化的子程序被重载,可以在实例化中包含一个签名来表明我们指的是哪个未实例化版本。例如: ```vhdl procedure combine generic ( type T ) parameter ( x : T; value : bit ); procedure combine generic ( type T ) parameter ( x : T; value : integer ); procedure combine_vec_with_bit is new combine[T, bit] generic map ( T => bit_vector ); ``` ##### 2.6 泛型映射子程序 和包一样,我们也可以在子程序中包含泛型映射。这样的子程序称为泛型映射子程序。 - **泛型映射过程语法**: ```plaintext subprogram_body ⇐ procedure identifier [ generic ( generic_interface_list ) [ generic map ( generic_association_list ) ] ] [ [ parameter ] ( parameter_interface_list ) ] is { subprogram_declarative_part } begin { sequential_statement } end [ procedure ] [ identifier ] ; ``` - **泛型映射函数语法**: ```plaintext subprogram_body ⇐ [ pure I impure ] function identifier [ generic ( generic_interface_list ) [ generic map ( generic_association_list ) ] ] [ [ parameter ] ( parameter_interface_list ) ] return type_mark is { subprogram_declarative_item } begin { sequential_statement } end [ function ] [ identifier ] ; ``` 通常我们不会显式编写泛型映射子程序,了解这个特性是为了在分析器产生看似神秘的错误消息时有所帮助。 #### 3. 泛型子程序 VHDL 允许声明泛型子程序,作为一种为单元提供要执行的操作或动作的方式。我们在泛型列表中声明一个形式泛型子程序,代表一个尚未指定的子程序,并在具有泛型列表的单元内包含对该形式泛型子程序的调用。当实例化该单元时,为该实例提供一个实际子程序。对形式泛型子程序的每次调用都代表对实例中实际子程序的调用。 ##### 3.1 形式泛型子程序声明语法 - **过程**: ```plaintext interface_declaration ⇐ procedure identifier [ [ parameter ] ( parameter_interface_list ) ] ``` - **函数**: ```plaintext interface_declaration ⇐ procedure identifier [ [ parameter ] ( parameter_interface_list ) ] return type_mark ``` 形式泛型子程序必须是一个简单的子程序,即它本身不能包含泛型列表。 ##### 3.2 示例:带增量函数的通用计数器 ```vhdl entity generic_counter is generic ( type count_type; constant reset_value : count_type; function increment ( x : count_type ) return count_type ); port ( clk, reset : in bit; data : out count_type ); end entity generic_counter; architecture rtl of generic_counter is begin count : process (clk) is begin if rising_edge(clk) then if reset then data <= reset_value; else data <= increment(data); end if; end if; end process count; end architecture rtl; use ieee.numeric_std.all; function add1 ( arg : unsigned ) return unsigned is begin return arg + 1; end function add1; signal clk, reset : bit; signal count_val : unsigned(15 downto 0); counter : entity work.generic_counter(rtl) generic map ( count_type => unsigned(15 downto 0), reset_value => (others => '0'), increment => add1 ) port map ( clk => clk, reset => reset, data => count_val ); ``` 我们还可以为不同类型实例化这个计数器,例如为 `traffic_light_color` 类型: ```vhdl type traffic_light_color is (red, yellow, green); function next_color ( arg : traffic_light_color ) return traffic_light_color is begin if arg = traffic_light_color'high then return traffic_light_color'low; else return traffic_light_color'succ(arg); end if; end function next_color; signal east_light : traffic_light_color; east_counter : work.generic_counter(rtl) generic map ( count_type => traffic_light_color, reset_value => red, increment => next_color ) port map ( clk => clk, reset => reset, data => east_light ); ``` ##### 3.3 默认形式泛型子程序 当在泛型列表中声明形式泛型子程序时,可以指定一个默认子程序,在没有为实例提供实际泛型子程序时使用。 - **带默认值的形式泛型过程声明**: ```plaintext interface_declaration ⇐ procedure identifier [ [ parameter ] ( parameter_interface_list ) ] is subprogram_name ``` - **带默认值的形式泛型函数声明**: ```plaintext interface_declaration ⇐ function identifier [ [ parameter ] ( parameter_interface_list ) ] return type_mark is subprogram_name ``` 所命名的子程序必须在该点可见。 ##### 3.4 示例:包中的通用错误报告 ```vhdl package error_utility_pkg is procedure report_error ( report_string : string; report_severity : severity_level ); end package error_utility_pkg; package body error_utility_pkg is procedure report_error ( report_string : string; report_severity : severity_level ) is begin report report_string severity report_severity; end procedure report_error; end package body error_utility_pkg; package operations is generic ( procedure error_action ( report_string : string; report_severity : severity_level ) is work.error_utility_pkg.report_error ); procedure step1 ( ... ); ... end package operations; package body operations is procedure step1 ( ... ) is begin ... if something_is_wrong then error_action("Something is wrong in step1", error); end if; ... end procedure step1; ... end package body operations; package reporting_operations is new work.operations; use re ```
corwn 最低0.47元/天 解锁专栏
赠100次下载
继续阅读 点击查看下一篇
profit 400次 会员资源下载次数
profit 300万+ 优质博客文章
profit 1000万+ 优质下载资源
profit 1000万+ 优质文库回答
复制全文

相关推荐

Big黄勇

硬件工程师
广州大学计算机硕士,硬件开发资深技术专家,拥有超过10多年的工作经验。曾就职于全球知名的大型科技公司,担任硬件工程师一职。任职期间负责产品的整体架构设计、电路设计、原型制作和测试验证工作。对硬件开发领域有着深入的理解和独到的见解。
最低0.47元/天 解锁专栏
赠100次下载
百万级 高质量VIP文章无限畅学
千万级 优质资源任意下载
千万级 优质文库回答免费看
立即解锁

专栏目录

最新推荐

【心电信号情绪识别在虚拟现实中的应用研究】:探索虚拟世界中的情绪分析

![【心电信号情绪识别在虚拟现实中的应用研究】:探索虚拟世界中的情绪分析](https://www.radsport-rennrad.de/wp-content/uploads/2018/10/leistungstest-radsport.jpg) # 摘要 情绪识别技术与虚拟现实的结合为沉浸式体验带来了新的可能性。本文首先概述了情绪识别与虚拟现实的基本概念,接着深入探讨了心电信号(ECG)的理论基础,包括其产生原理、采集方法和数据处理技术。文中详细分析了心电信号情绪识别算法,并研究了机器学习和深度学习在情绪识别中的应用。此外,本文还探讨了心电信号情绪识别技术在虚拟现实中的实际应用,并通过具

地震波正演中的不确定性分析:识别与减少模拟误差的专业方法

![吸收边界](https://media.springernature.com/lw1200/springer-static/image/art%3A10.1007%2Fs42114-022-00514-2/MediaObjects/42114_2022_514_Fig1_HTML.png) # 摘要 地震波正演模拟是地震学研究中的重要工具,它能够模拟波在地下介质中的传播过程,并用于解释和预测实际地震数据。本文首先介绍地震波正演模拟的基础知识,然后详细探讨了地震波正演模拟中存在的不确定性因素,包括地质模型和物理参数的不确定性,并分析了识别和量化这些不确定性的方法。接着,本文探讨了减少正演模

【飞机缺陷实时检测系统构建】:挑战与策略并重

![【飞机缺陷实时检测系统构建】:挑战与策略并重](https://img-blog.csdnimg.cn/a30e05f512b04c9686b67052dacd8bae.png) # 摘要 飞机缺陷实时检测系统是确保航空安全和提升维护效率的关键技术。本文首先阐述了系统的基本概念和重要性,接着探讨了实时检测技术的理论基础,包括图像处理技术、机器学习及深度学习的应用,以及实时数据流处理技术的挑战与方法。第三章介绍了系统构建的实践过程,涵盖了系统设计、关键技术实现以及系统测试与优化。第四章着重讨论了系统的安全与维护策略,包括数据安全、系统防护机制以及维护与升级流程。第五章通过案例分析,讨论了成

【多源数据整合王】:DayDreamInGIS_Geometry在不同GIS格式中的转换技巧,轻松转换

![【多源数据整合王】:DayDreamInGIS_Geometry在不同GIS格式中的转换技巧,轻松转换](https://community.esri.com/t5/image/serverpage/image-id/26124i748BE03C6A81111E?v=v2) # 摘要 本论文详细介绍了DayDreamInGIS_Geometry这一GIS数据处理工具,阐述了其核心功能以及与GIS数据格式转换相关的理论基础。通过分析不同的GIS数据格式,并提供详尽的转换技巧和实践应用案例,本文旨在指导用户高效地进行数据格式转换,并解决转换过程中遇到的问题。文中还探讨了转换过程中的高级技巧、

手机Modem协议在网络环境下的表现:分析与优化之道

![手机Modem协议开发快速上手.docx](https://img-blog.csdnimg.cn/0b64ecd8ef6b4f50a190aadb6e17f838.JPG?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBATlVBQeiInOWTpQ==,size_20,color_FFFFFF,t_70,g_se,x_16) # 摘要 Modem协议在网络通信中扮演着至关重要的角色,它不仅定义了数据传输的基础结构,还涉及到信号调制、通信流程及错误检测与纠正机制。本文首先介

【C#数据绑定高级教程】:深入ListView数据源绑定,解锁数据处理新技能

![技术专有名词:ListView](https://androidknowledge.com/wp-content/uploads/2023/01/customlistthumb-1024x576.png) # 摘要 随着应用程序开发的复杂性增加,数据绑定技术在C#开发中扮演了关键角色,尤其在UI组件如ListView控件中。本文从基础到高级技巧,全面介绍了C#数据绑定的概念、原理及应用。首先概述了C#中数据绑定的基本概念和ListView控件的基础结构,然后深入探讨了数据源绑定的实战技巧,包括绑定简单和复杂数据源、数据源更新同步等。此外,文章还涉及了高级技巧,如数据模板自定义渲染、选中项

零信任架构的IoT应用:端到端安全认证技术详解

![零信任架构的IoT应用:端到端安全认证技术详解](https://img-blog.csdnimg.cn/20210321210025683.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQyMzI4MjI4,size_16,color_FFFFFF,t_70) # 摘要 随着物联网(IoT)设备的广泛应用,其安全问题逐渐成为研究的焦点。本文旨在探讨零信任架构下的IoT安全认证问题,首先概述零信任架构的基本概念及其对Io

物联网技术:共享电动车连接与控制的未来趋势

![物联网技术:共享电动车连接与控制的未来趋势](https://read.nxtbook.com/ieee/potentials/january_february_2020/assets/4cf66356268e356a72e7e1d0d1ae0d88.jpg) # 摘要 本文综述了物联网技术在共享电动车领域的应用,探讨了核心的物联网连接技术、控制技术、安全机制、网络架构设计以及实践案例。文章首先介绍了物联网技术及其在共享电动车中的应用概况,接着深入分析了物联网通信协议的选择、安全机制、网络架构设计。第三章围绕共享电动车的控制技术,讨论了智能控制系统原理、远程控制技术以及自动调度与充电管理

【仿真模型数字化转换】:从模拟到数字的精准与效率提升

![【仿真模型数字化转换】:从模拟到数字的精准与效率提升](https://img-blog.csdnimg.cn/42826d38e43b44bc906b69e92fa19d1b.png) # 摘要 本文全面介绍了仿真模型数字化转换的关键概念、理论基础、技术框架及其在实践中的应用流程。通过对数字化转换过程中的基本理论、关键技术、工具和平台的深入探讨,文章进一步阐述了在工程和科学研究领域中仿真模型的应用案例。此外,文中还提出了数字化转换过程中的性能优化策略,包括性能评估方法和优化策略与方法,并讨论了数字化转换面临的挑战、未来发展趋势和对行业的长远意义。本文旨在为专业人士提供一份关于仿真模型数

虚拟助理引领智能服务:酒店行业的未来篇章

![虚拟助理引领智能服务:酒店行业的未来篇章](https://images.squarespace-cdn.com/content/v1/5936700d59cc68f898564990/1497444125228-M6OT9CELKKA9TKV7SU1H/image-asset.png) # 摘要 随着人工智能技术的发展,智能服务在酒店行业迅速崛起,其中虚拟助理技术在改善客户体验、优化运营效率等方面起到了关键作用。本文系统地阐述了虚拟助理的定义、功能、工作原理及其对酒店行业的影响。通过分析实践案例,探讨了虚拟助理在酒店行业的应用,包括智能客服、客房服务智能化和后勤管理自动化等方面。同时,