单片机寄存器相关知识及应用(51单片机)
在前面的STM32中我并没有直接对寄存器进行操作,而是通过固件库直接引用进行各个外设的配置和应用,现在,我开始进行寄存器的学习(51单片机)。
我们先简单看一下80C51/52的微控制头文件 <REG52.h>
一、字节寄存器定义
- 定义了一系列的特殊功能寄存器,如 P0、P1、P2、P3、PSW、ACC、B、SP、DPL、DPH、PCON、TCON、TMOD、TL0、TL1、TH0、TH1、IE、IP、SCON、SBUF。
- 对于 8052 扩展部分,定义了 T2CON、RCAP2L、RCAP2H、TL2、TH2。
二、位寄存器定义
- PSW 相关位
- 定义了 PSW 寄存器中的各个位,如 CY(进位标志位)、AC(辅助进位标志位)、F0(用户标志位)、RS1 和 RS0(寄存器组选择位)、OV(溢出标志位)、P(8052 特有的奇偶校验位)。
- TCON 相关位
- 定义了 TCON 寄存器中的各个位,如 TF1(定时器 1 溢出标志位)、TR1(定时器 1 运行控制位)、TF0、TR0、IE1、IT1、IE0、IT0。
- IE 相关位
- 定义了 IE 寄存器中的各个位,如 EA(总中断允许位)、ET2(8052 特有的定时器 2 中断允许位)、ES、ET1、EX1、ET0、EX0。
- IP 相关位
- 定义了 IP 寄存器中的各个位,如 PT2、PS、PT1、PX1、PT0、PX0。
- P3 相关位
- 定义了 P3 端口的各个位,如 RD(外部数据存储器读选通位)、WR、T1、T0、INT1、INT0、TXD、RXD。
- SCON 相关位
- 定义了 SCON 寄存器中的各个位,如 SM0、SM1、SM2、REN、TB8、RB8、TI、RI。
- P1 相关位(8052 特有的)
- 定义了 P1 端口的两个位 T2EX 和 T2。
- T2CON 相关位
- 定义了 T2CON 寄存器中的各个位,如 TF2、EXF2、RCLK、TCLK、EXEN2、TR2、C_T2、CP_RL2。
这个微控制器有四个端口,分别是:
- P0:8 位并行输入 / 输出端口。
- P1:8 位并行输入 / 输出端口。
- P2:8 位并行输入 / 输出端口。
- P3:8 位并行输入 / 输出端口,同时部分引脚具有第二功能。
比如我们要操作一个蜂鸣器或者一个LED灯的亮暗,我们就可以查看想要在那个引脚处进行驱动,执行高低电平,这里是控制一个低电平触发的蜂鸣器。
sbit等价于set bit,也就是设置引脚的作用,在这个程序中我们设置了P1端口的第1位,并设置为低电平态。
重温继电器:
- 继电器内部主要由线圈和触点组成。当线圈两端加上合适的电压时,会产生磁场,使触点状态发生改变。在低电平触发的情况下,线圈通过低电平信号驱动电流,从而产生磁场使继电器动作。
- 继电器的触点可以用于控制外部电路的通断,例如连接高电压、大电流的负载。
使用继电器实现对LED灯的控制
我们如果想写一个延迟的函数可以借助STC—ISP下载器,在这里可以直接生成相对应的代码。
利用震动传感器实现震动感应灯
注:震动传感器为低电平触发
中断知识:
在 51 单片机中,中断是一种重要的机制,它允许单片机在执行主程序的过程中,暂停当前任务,转而去处理更紧急的事件,处理完后再返回原来被中断的地方继续执行主程序。
一、中断的作用
- 提高系统的实时性:当有紧急事件发生时,能够及时响应,而不必等待主程序循环执行到相应的位置。比如,外部设备发送数据需要立即处理,通过中断可以快速响应,避免数据丢失。
- 实现多任务处理:在单一处理器上模拟多任务的效果。不同的中断源可以代表不同的任务,单片机可以根据中断的优先级依次处理这些任务。
二、中断源
51 单片机通常有多个中断源,主要包括:
- 外部中断:由外部信号触发,可以设置为低电平触发或下降沿触发。例如,按键按下时产生的外部中断信号。
- 定时器中断:当定时器计数溢出时产生中断。可以用于定时控制、产生周期性的信号等。
- 串口中断:当串口接收到数据或发送完数据时产生中断,方便进行数据的接收和发送处理。
三、中断优先级
51 单片机具有两级中断优先级,可以设置高优先级和低优先级中断。当多个中断同时发生时,高优先级的中断会先被响应。如果同一优先级的中断同时发生,则按照硬件查询顺序响应。
四、中断的过程
- 中断请求:当中断源产生中断信号时,向单片机的中断系统发出中断请求。
- 中断响应:如果中断允许标志位(IE 寄存器中的相应位)被置位,并且没有更高优先级的中断正在被处理,单片机就会响应中断。响应中断时,单片机首先完成当前指令的执行,然后将程序计数器(PC)的值压入堆栈保存,接着将相应的中断入口地址装入 PC,开始执行中断服务程序。
- 中断服务:在中断服务程序中,处理中断源对应的事件。这是中断处理的核心部分,根据具体的需求编写相应的代码。
- 中断返回:中断服务程序执行完毕后,通过执行中断返回指令(RETI),将堆栈中保存的程序计数器的值弹出,恢复到中断前的程序执行位置,继续执行主程序。
中断系统:
8051 单片机的中断控制系统工作程序主要包括以下几个步骤:
一、中断源及相关寄存器设置
8051 单片机有 5 个中断源,分别是外部中断 0(INT0)、外部中断 1(INT1)、定时器 / 计数器 0 溢出中断(T0)、定时器 / 计数器 1 溢出中断(T1)和串行口中断。每个中断源都有相应的控制寄存器来设置其触发方式、使能状态等。
-
中断允许寄存器 IE(Interrupt Enable):
- EA:总中断允许位,EA = 1 时,开放所有中断;EA = 0 时,关闭所有中断。
- EX0 和 EX1:分别为外部中断 0 和外部中断 1 的允许位。
- ET0 和 ET1:分别为定时器 / 计数器 0 和定时器 / 计数器 1 溢出中断允许位。
- ES:串行口中断允许位。
-
中断优先级寄存器 IP(Interrupt Priority):
- PX0 和 PX1:分别为外部中断 0 和外部中断 1 的优先级控制位。
- PT0 和 PT1:分别为定时器 / 计数器 0 和定时器 / 计数器 1 溢出中断的优先级控制位。
- PS:串行口中断的优先级控制位。
二、中断触发方式设置
-
外部中断触发方式:
- 通过定时器控制寄存器 TCON 中的 IT0 和 IT1 位来设置外部中断 0 和外部中断 1 的触发方式。
- IT0 = 0 时,外部中断 0 为低电平触发;IT0 = 1 时,外部中断 0 为下降沿触发。
- IT1 同理。
-
定时器 / 计数器溢出中断:由定时器 / 计数器的计数溢出自动产生中断请求。
三、中断响应过程
-
中断请求:当某个中断源满足触发条件时,就会向 CPU 发出中断请求。例如,外部中断源在触发条件满足时,将相应的中断请求标志位置位。
-
中断查询:CPU 在每个机器周期的最后一个状态(S6),查询所有中断源的中断请求标志位。如果有中断请求且相应的中断允许位为 1,则进入下一步。
-
中断响应:
- CPU 响应中断后,首先自动执行一条硬件指令 “LCALL” 或 “LJMP”,将程序计数器 PC 的内容压入堆栈保护,然后将相应的中断入口地址装入 PC,转去执行中断服务程序。
- 8051 单片机的 5 个中断源分别对应 5 个固定的中断入口地址:外部中断 0 为 0003H,外部中断 1 为 0013H,定时器 / 计数器 0 溢出中断为 000BH,定时器 / 计数器 1 溢出中断为 001BH,串行口中断为 0023H。
-
中断服务程序执行:在中断服务程序中,根据具体的中断源进行相应的处理操作。例如,对于外部中断,可以读取外部信号状态进行相应的处理;对于定时器中断,可以进行定时操作或计数操作等。
-
中断返回:中断服务程序执行完毕后,执行中断返回指令 “RETI”。该指令将堆栈中保存的程序计数器 PC 的值弹出,使程序返回到被中断的地方继续执行。
四、中断优先级处理
当多个中断源同时发出中断请求时,CPU 按照中断优先级进行响应。8051 单片机的中断优先级分为两级,高优先级中断可以中断低优先级中断的服务程序。如果同一优先级的多个中断源同时请求中断,CPU 将按照自然优先级顺序进行响应,其顺序为外部中断 0、定时器 / 计数器 0 溢出中断、外部中断 1、定时器 / 计数器 1 溢出中断、串行口中断。
芯片图:
使用中断:
利用中断实现震动感应灯
注:利用中断可节约CPU资源
#include <REG52.h>
#include <INTRINS.H>
sbit LED = P1^1;
sbit SHARK = P1^0;
void Delay500ms() //@11.0592MHz
{
unsigned char i, j, k;
_nop_();
_nop_();
i = 22;
j = 3;
k = 227;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
void init()
{
EA = 1;//¿ªÆô×ÜÖжÏ
EX0 =1;//¿ªÆôEX0µÄÍⲿÖжÏ
IT0=1;//µÍµçƽ´¥·¢·½Ê½
}
int main()
{
init();
while(1)
{
}
return 0;
}
void int0() interrupt 0
{
LED = 0;
Delay500ms();
LED=1;
Delay500ms();
}
定时器相关知识:
一、定时器的作用
- 定时功能:可以实现精确的时间延迟,用于控制程序中的时间间隔,例如周期性地执行某些任务、产生定时中断等。
- 计数功能:对外部脉冲进行计数,可以用于测量外部事件的频率、脉冲宽度等。
二、定时器的结构
8051 单片机通常有两个定时器 / 计数器:T0 和 T1。每个定时器都由以下部分组成:
- 计数器:16 位的计数器,可以对内部机器周期进行计数(定时模式)或对外部脉冲进行计数(计数模式)。
- 工作方式寄存器:用于设置定时器的工作方式,如定时 / 计数模式选择、计数位数等。
- 控制寄存器:控制定时器的启动、停止、中断等操作。
三、工作方式
- 方式 0:13 位定时器 / 计数器。由 THx 的高 8 位和 TLx 的低 5 位组成 13 位计数器。
- 方式 1:16 位定时器 / 计数器。THx 和 TLx 组成 16 位计数器。
- 方式 2:8 位自动重装定时器 / 计数器。TLx 作为计数器,当计数溢出时,自动将 THx 的内容重装到 TLx 中。
- 方式 3:T0 分为两个独立的 8 位定时器 / 计数器,T1 停止计数。
四、定时时间的计算
定时时间取决于计数脉冲的频率和计数器的初值。
- 定时模式下,计数脉冲的频率是单片机的时钟频率经过 12 分频得到的。若单片机时钟频率为 fosc,则计数脉冲频率为 fosc/12。
- 定时时间 T = (2^n - 计数初值)× 机器周期。其中,n 为计数器的位数(方式 0 时 n = 13,方式 1 时 n = 16,方式 2 时 n = 8),机器周期 = 12 /fosc。
五、定时器的编程步骤
- 设置定时器的工作方式:通过对 TMOD 寄存器进行赋值来选择定时器的工作方式、定时 / 计数模式等。
- 计算计数初值:根据所需的定时时间,计算出计数器的初值,并分别写入 THx 和 TLx 寄存器。
- 启动定时器:将 TRx 位置 1,启动定时器开始计数。
- 处理定时器中断(如果需要):当定时器计数溢出时,会产生中断请求。可以在中断服务程序中处理定时任务。
六、应用场景
- 实现精确的时间延迟,如控制 LED 的闪烁频率、电机的转速等。
- 作为实时时钟的时间基准,实现时钟功能。
- 测量外部信号的频率、脉冲宽度等。
- 在多任务系统中,作为时间片轮转调度的时钟源。
定时器相关知识:
定时器用于计数系统,可实现软件计时,使程序每隔一段时间完成一项操作
大部分用于替代在程序中长时间delay(函数),从而提高CPU的运行效率和处理速度
STC89C52有3个定时器(T0,T1,T2),这比传统的51单片机多一个T2
该定时器 / 计数器 0 处于模式 1,即 16 位定时器 / 计数器模式。在这种模式下,它可以实现较为精确的定时功能或者对外部事件进行计数。
SYSclk表示系统时钟,定时器和计数器都来自系统时钟,选择不同的时钟模式可以调整计数速度,12T和6T。
12T表示一个机器周期由12个时钟周期组成,6T表示一个机器周期由6个时钟周期组成,所以6T计数速度更快,频率更高,精度更高,但会消耗更多的功率。
“TRO C/T=0 control (8 Bits)”、“TLO (8 bits)” 和 “THO” 分别对应定时器 / 计数器的低 8 位控制寄存器、低 8 位计数器和高 8 位计数器。通过对这些寄存器的设置,可以控制定时器 / 计数器的工作方式,如启动、停止、复位等。
“TFO Interrupt” 表示定时器 / 计数器溢出时可以产生中断。当定时器 / 计数器从初始值计数到最大值后,会产生溢出标志,并可以触发中断请求,以便 CPU 进行相应的处理。
“GATE” 和 “INTO” 可能与外部控制信号和中断输入有关。例如,GATE 可以用于控制定时器 / 计数器的启动和停止是否受外部信号的影响,INTO 可能是外部中断输入引脚。
一、模式 0:13 位定时器 / 计数器
在模式 0 下,定时器 / 计数器被配置为 13 位的计数器。其中,低 5 位由 TLO(定时器 / 计数器低 5 位寄存器)组成,高 8 位由 THO(定时器 / 计数器高 8 位寄存器)的低 5 位组成。
这种模式的计数范围相对较小,但在一些对定时精度要求不高且需要较小计数范围的应用中可能会有用。例如,简单的定时闪烁小灯,或者作为一个简单的时间间隔测量工具。
二、模式 1:16 位定时器 / 计数器(常用)
这是一种较为常用的模式。16 位的定时器 / 计数器由 TLO(低 8 位)和 THO(高 8 位)组成。
它具有较大的计数范围,可以满足大多数一般应用的定时和计数需求。比如,可以用于产生一定时间间隔的中断,以实现周期性的任务执行;或者用于对外部脉冲进行计数,以测量外部事件的发生频率等。
三、模式 2:8 位自动重装载模式
在模式 2 下,定时器 / 计数器被配置为 8 位的计数器。当计数器计满溢出时,自动将一个预先设置好的值重新装载到计数器中,开始下一轮计数。
这种模式的优点是可以在不需要软件干预的情况下实现精确的定时。例如,在需要周期性地产生一个固定时间间隔的中断时,可以使用模式 2。设置好自动重装载的值后,定时器 / 计数器会自动重复计数,无需在每次溢出后通过软件重新设置计数值。
四、模式 3:两个 8 位计数器
在模式 3 下,定时器 / 计数器 0 被分成两个独立的 8 位计数器。其中,TL0 作为一个 8 位计数器使用,而 TH0 则被固定为一个 8 位定时器,只能用于定时功能,不能作为计数器使用。定时器 / 计数器 1 在模式 3 下停止计数。
这种模式在一些特殊的应用场景中可能会有用。例如,当需要同时进行两个不同时间间隔的定时任务,且其中一个任务对定时精度要求较高时,可以使用模式 3 中的 TH0 作为高精度定时器,而 TL0 则用于另一个定时任务。
1.寄存器是连接软硬件的媒介
2.在单片机中寄存器就是一段特殊的RAM存储器,一方面,寄存器可以存储和读取数据,另一方面,每一个寄存器背后都连接了一根导线,控制着电路的连接方式。
3.寄存器相当于一个复杂机器的”操作按钮”
利用定时器控制灯
#include <REG52.h>
#include <INTRINS.H>
sbit LED = P1^1;
sbit SHARK = P1^0;
// 定义一个全局变量 T0count,用于记录定时器 0 的中断次数
unsigned int T0count;
// 定时器初始化函数
void timer_init()
{
// 设置定时器 0 的工作模式为模式 1(16 位定时器)
TMOD = 0X01;
// 清除定时器 0 的溢出标志位
TF0 = 0;
// 启动定时器 0
TR0 = 1;
// 给定时器 0 的高 8 位赋初值,64535/256
TH0 = 64535/256;
// 给定时器 0 的低 8 位赋初值,64535%256
TL0 = 64535%256;
// 使能定时器 0 中断
ET0 = 1;
// 使能总中断
EA = 1;
// 设置定时器 0 中断为高优先级
PT0 = 1;
}
// 软件延时函数,延时约 500 毫秒(在 11.0592MHz 晶振下)
void Delay500ms() //@11.0592MHz
{
unsigned char i, j, k;
// 空操作指令,用于稍微延迟一点时间(可能是为了确保稳定或其他未知目的)
_nop_();
_nop_();
// 初始化变量 i、j、k
i = 22;
j = 3;
k = 227;
// 三层嵌套循环实现延时
do
{
do
{
// k 递减直到为 0
while (--k);
} while (--j);
} while (--i);
}
// 主函数
void main()
{
// 调用定时器初始化函数
timer_init();
while(1)
{
// 主循环中目前没有任何操作,只是空循环等待中断发生
}
}
// 定时器 0 中断服务函数
void timer0_routine() interrupt 1
{
// 重新给定时器 0 的高 8 位赋初值,64535/256
TH0 = 64535/256;
// 重新给定时器 0 的低 8 位赋初值,64535%256
TL0 = 64535%256;
// 定时器 0 的中断次数加一
T0count++;
// 当定时器 0 的中断次数达到 1000 时
if(T0count>=1000)
{
// 将 LED 引脚置为低电平,点亮 LED
LED = 0;
// 调用软件延时函数,延时约 500 毫秒
Delay500ms();
// 将 LED 引脚置为高电平,熄灭 LED
LED = 1;
// 再次调用软件延时函数,延时约 500 毫秒
Delay500ms();
// 将中断次数重置为 0
T0count = 0;
}
}
对于定时器初始化函数timer_init();
-
TMOD = 0X01;
TMOD
是定时器 / 计数器的工作方式寄存器。- 设置
TMOD
为0X01
,表示将定时器 0 设置为工作模式 1,即 16 位定时器模式。在这种模式下,定时器由高 8 位和低 8 位两个寄存器组成,计数范围为 0 到 65535。
-
TF0=0;
TF0
是定时器 0 的溢出标志位。- 将其初始化为 0,表示当前定时器 0 没有发生溢出。
-
TR0=1;
TR0
是定时器 0 的运行控制位。- 设置为 1 表示启动定时器 0,开始计数。
-
TH0=64535/256;
和TL0=64535%256;
TH0
是定时器 0 的高 8 位计数器初始值,TL0
是低 8 位计数器初始值。- 这里将初始值设置为 64535,即当定时器从这个值开始计数,经过一定时间后会产生溢出中断。具体的计数时间取决于晶振频率和定时器的工作模式。
- 这两个值其实是由2^16-1=64535,TH0=64535/256(2^8)可取得高八位,同理可取得低8位。
-
ET0=1;
ET0
是定时器 0 的中断允许位。- 设置为 1 表示允许定时器 0 产生中断。
-
EA=1;
EA
是总中断允许位。- 设置为 1 表示允许所有中断,包括定时器 0 的中断。
-
PT0=1;
PT0
是定时器 0 的中断优先级控制位。- 设置为 1 表示将定时器 0 的中断设置为高优先级。
综上所述,这个定时器初始化函数的作用是配置定时器 0 为 16 位定时器模式,设置初始计数值,允许中断并启动定时器,同时将定时器 0 的中断设置为高优先级。这样,定时器 0 就可以按照设定的方式进行计数,并在满足条件时产生中断,触发相应的中断服务程序。
串口相关:
这里我只叙述一下前面32单片机没有讲到的知识点。
一、SBUF(串行口缓冲寄存器)
SBUF 在发送和接收过程中起到关键作用。在发送时,写 SBUF 意味着将数据写入发送缓冲寄存器(通常与 TH1、TL1 及发送控制器等相关联),然后数据通过发送控制器和控制门,从 TXD(发送数据引脚)发送出去。在接收时,数据从 RXD(接收数据引脚)进入,经过接收控制器存入 SBUF,此时读 SBUF 可以获取接收到的数据。
二、发送过程
- 当向 SBUF 写入数据后,数据会进入发送控制器,同时可能与 TH1 和 TL1(通常与定时器 1 相关)以及 T1 溢出率有关。T1 溢出率可能会影响发送的速率等特性。
- 发送控制器根据控制信号和相关设置,将数据通过控制门从 TXD 引脚发送出去。当发送完成后,会触发 TI(发送中断标志位),可以去中断逻辑进行相应处理。
三、接收过程
- 数据从 RXD 引脚接收进来,经过接收控制器存入 SBUF。接收控制器可能会根据一些控制信号如 SMOD(电源控制寄存器的双倍波特率位)等进行相应的操作。
- 当读取 SBUF 时,可以获取接收到的数据。
四、控制门及其他设置
- 控制门在发送和接收过程中起到数据通路的控制作用,确保数据在正确的时间和方向上传输。
一、中断系统概述
中断系统是单片机中非常重要的部分,它允许单片机在执行主程序的过程中,暂停当前任务去响应紧急事件或外部请求,处理完中断事件后再返回主程序继续执行。
二、各个部分的作用
- 控制门 TX 和收控利(接收控制):控制数据的发送和接收,确保数据在正确的时间和方向上传输。
- 位寄存:可能用于存储特定的状态位或数据位,以便在中断处理过程中进行判断和操作。
- TCON(定时器 / 计数器控制寄存器)和 SCON(串行口控制寄存器):
- TCON 主要用于控制定时器 / 计数器的启动、停止,以及设置中断标志等。例如,IE(中断允许寄存器)和 IP(中断优先级寄存器)可以设置各个中断源的允许和优先级。
- SCON 主要用于控制串行口的工作方式、接收和发送状态等。
- 硬件查询:可以通过查询特定的寄存器状态来判断是否有中断请求发生。
- 中断优先级:由 ITO(外部中断 0 触发方式控制位)、EXO(外部中断 0 允许位)、PX(外部中断 0 优先级控制位)等控制外部中断 0 的优先级;同理, INTO(外部中断 1 触发方式控制位)、IEO(外部中断 1 允许位)等控制外部中断 1 的优先级。高级中断源具有更高的优先级,在多个中断同时发生时,优先级高的先被响应。
- 中断入口:当中断发生时,程序会跳转到相应的中断入口地址去执行中断服务程序。例如,TO(定时器 / 计数器 T0 溢出中断请求标志位)、T1(定时器 / 计数器 T1 溢出中断请求标志位)、TI(串行口发送中断请求标志位)、TF(串行口接收中断请求标志位)等中断标志位被置位时,会触发相应的中断入口。
- 中断激发:外部事件或内部定时器 / 计数器溢出等情况可以激发中断请求。..
配置串口:
#include <REG52.h>
#include <INTRINS.H>
//#include "timer.h"
// 定义连接 LED 的引脚为 P1.1
sbit LED = P1^1;
// 定义连接 SHARK(假设是某个外部设备)的引脚为 P1.0
sbit SHARK = P1^0;
// 延时 500 毫秒的函数,在 11.0592MHz 晶振频率下
void Delay500ms() //@11.0592MHz
{
unsigned char i, j, k;
_nop_();
_nop_();
// 设置三层循环进行延时
i = 22;
j = 3;
k = 227;
do
{
do
{
while (--k);
} while (--j);
} while (--i);
}
// 串口初始化函数
void uartinit()
{
// 设置串口控制寄存器 SCON 为只允许发送模式
SCON=0x40;
// 设置电源控制寄存器 PCON 的最高位,即 SMOD 位,用于波特率加倍等特殊用途
PCON |= 0x80;
// 清除 TMOD 的高四位,保留低四位
TMOD &= 0X0F;
// 设置 TMOD 的高四位为 0010,即定时器 1 工作在模式 2(8 位自动重装模式)
TMOD |=0X20;
// 启动定时器 1
TR1=1;
// 设置定时器 1 的低 8 位初值为 0xF4
TL1 = 0xF4;
// 设置定时器 1 的高 8 位初值也为 0xF4,用于产生特定的波特率
TH1 = 0xF4;
// 禁止定时器 1 中断
ET1=0;
}
// 串口发送一个字节数据的函数
void UART_sendbit(unsigned char Byte)
{
// 将待发送的字节放入发送缓冲寄存器 SBUF
SBUF=Byte;
// 等待发送完成标志位 TI 置位
while(TI==0);
// 手动将 TI 标志位清零,以便下一次发送
TI=0;
}
// 主函数
void main()
{
// 初始化串口
uartinit();
// 发送十六进制数据 0x66
UART_sendbit(0x66);
while(1)
{
// 无限循环,程序在此处停止,等待外部事件
}
}