将“渴望“乐谱写入AT24C02并读出播放

#include <reg51.h>         // 包含51单片机寄存器定义的头文件
#include <intrins.h>       //包含_nop_()函数定义的头文件
#define    OP_READ    0xa1        // 器件地址以及读取操作,0xa1即为1010 0001B
#define    OP_WRITE 0xa0        // 器件地址以及写入操作,0xa1即为1010 0000B
sbit SDA=P3^4;             //将串行数据总线SDA位定义在为P3.4引脚
sbit SCL=P3^3;             //将串行时钟总线SDA位定义在为P3.3引脚
sbit sound=P3^7;           //将sound位定义为P3.7,从该引脚输出音频
unsigned int C;            //储存定时器的定时常数

//以下是C调低音的音频宏定义
#define l_dao 262   //将“l_dao”宏定义为低音“1”的频率262Hz
#define l_re 286    //将“l_re”宏定义为低音“2”的频率286Hz
#define l_mi 311    //将“l_mi”宏定义为低音“3”的频率311Hz
#define l_fa 349    //将“l_fa”宏定义为低音“4”的频率349Hz
#define l_sao 392   //将“l_sao”宏定义为低音“5”的频率392Hz
#define l_la 440    //将“l_a”宏定义为低音“6”的频率440Hz
#define l_xi 494    //将“l_xi”宏定义为低音“7”的频率494Hz
//以下是C调中音的音频宏定义
#define dao 523     //将“dao”宏定义为中音“1”的频率523Hz
#define re 587      //将“re”宏定义为中音“2”的频率587Hz
#define mi 659      //将“mi”宏定义为中音“3”的频率659Hz
#define fa 698      //将“fa”宏定义为中音“4”的频率698Hz
#define sao 784     //将“sao”宏定义为中音“5”的频率784Hz
#define la 880      //将“la”宏定义为中音“6”的频率880Hz
#define xi 987      //将“xi”宏定义为中音“7”的频率523Hz
//以下是C调高音的音频宏定义
#define h_dao 1046     //将“h_dao”宏定义为高音“1”的频率1046Hz
#define h_re 1174      //将“h_re”宏定义为高音“2”的频率1174Hz
#define h_mi 1318      //将“h_mi”宏定义为高音“3”的频率1318Hz
#define h_fa 1396     //将“h_fa”宏定义为高音“4”的频率1396Hz
#define h_sao 1567    //将“h_sao”宏定义为高音“5”的频率1567Hz
#define h_la 1760     //将“h_la”宏定义为高音“6”的频率1760Hz
#define h_xi 1975     //将“h_xi”宏定义为高音“7”的频率1975Hz
/*******************************************
函数功能:节拍的延时的基本单位,延时200ms
******************************************/
void delay()               
   {
     unsigned char i,j;
      for(i=0;i<250;i++)
        for(j=0;j<250;j++)
                 ;
   }
/**************************************************************************
以下是对AT24C02进行读写操作的源程序
 *************************************************************************/
/*****************************************************
函数功能:延时1ms
(3j+2)*i=(3×33+2)×10=1010(微秒),可以认为是1毫秒
***************************************************/
void delay1ms()
{
   unsigned char i,j;    
     for(i=0;i<10;i++)
      for(j=0;j<33;j++)
            ;         
 }
/*****************************************************
函数功能:延时若干毫秒
入口参数:n
***************************************************/
 void delaynms(unsigned char n)
 {
   unsigned char i;
    for(i=0;i<n;i++)
       delay1ms();
 }
/***************************************************
函数功能:开始数据传送
***************************************************/
void start()
{
    SDA = 1;     //SDA初始化为高电平"1"
    SCL = 1;     //开始数据传送时,要求SCL为高电平"1"
    _nop_();    //等待一个机器周期
    _nop_();    //等待一个机器周期
    SDA = 0;     //SDA的下降沿被认为是开始信号
    _nop_();    //等待一个机器周期
    _nop_();    //等待一个机器周期
    _nop_();    //等待一个机器周期
    _nop_();    //等待一个机器周期
    SCL = 0;     //SCL为低电平时,SDA上数据才允许变化(即允许以后的数据传递)
}
/***************************************************
函数功能:结束数据传送
***************************************************/
void stop()
{
    SDA = 0;      //SDA初始化为低电平"0"
    _nop_();     //等待一个机器周期
    _nop_();     //等待一个机器周期
    SCL = 1;      //结束数据传送时,要求SCL为高电平"1"
    _nop_();     //等待一个机器周期
    _nop_();     //等待一个机器周期
    _nop_();     //等待一个机器周期
    _nop_();     //等待一个机器周期
    SDA = 1;     //SDA的上升沿被认为是结束信号
}
/***************************************************
函数功能:从AT24Cxx读取数据
出口参数:x
***************************************************/
unsigned char ReadData()
{
    unsigned char i;
    unsigned char x;               //储存从AT24Cxx中读出的数据
    for(i = 0;i < 8;i++)
    {
        SCL = 1;                 //SCL置为高电平
        x<<=1;                  //将x中的各二进位向左移一位
        x|=(unsigned char)SDA; //将SDA上的数据通过按位"或"运算存入x中
        SCL = 0;                //在SCL的下降沿读出数据
    }
    return(x);                  //将读取的数据返回
}
/***************************************************
函数功能:向AT24Cxx的当前地址写入数据
入口参数:y (储存待写入的数据)
***************************************************/
//在调用此数据写入函数前需首先调用开始函数start(),所以SCL=0
bit WriteCurrent(unsigned char y)
{
    unsigned char i;
    bit ack_bit;                  //储存应答位
    for(i = 0; i < 8; i++)         // 循环移入8个位
    {
        SDA = (bit)(y&0x80); //通过按位"与"运算将最高位数据送到S
                                //因为传送时高位在前,低位在后
        _nop_();              //等待一个机器周期   
       SCL = 1;                //在SCL的上升沿将数据写入AT24Cxx      
          _nop_();              //等待一个机器周期 
       _nop_();              //等待一个机器周期       
       SCL = 0;  //将SCL重新置为低电平,以在SCL线形成传送数据所需的8个脉冲
        y <<= 1;             //将y中的各二进位向左移一位
    }
    SDA = 1; // 发送设备(主机)应在时钟脉冲的高电平期间(SCL=1)释放SDA线,
              //以让SDA线转由接收设备(AT24Cxx)控制
    _nop_();     //等待一个机器周期 
    _nop_();     //等待一个机器周期 
    SCL = 1;      //根据上述规定,SCL应为高电平
    _nop_();     //等待一个机器周期 
    _nop_();     //等待一个机器周期 
    _nop_();     //等待一个机器周期 
    _nop_();     //等待一个机器周期 
    ack_bit = SDA; //接受设备(AT24Cxx)向SDA送低电平,表示已经接收到一个字节
                   //若送高电平,表示没有接收到,传送异常
    SCL = 0;      //SCL为低电平时,SDA上数据才允许变化(即允许以后的数据传递)
    return  ack_bit;// 返回AT24Cxx应答位
}
/***************************************************
函数功能:向AT24Cxx中的指定地址写入数据
入口参数:add (储存指定的地址);dat(储存待写入的数据)
***************************************************/
void WriteSet(unsigned char add, unsigned char dat)
{
    start();                   //开始数据传递
    WriteCurrent(OP_WRITE);  //选择要操作的AT24Cxx芯片,并告知要对其写入数据
    WriteCurrent(add);        //写入指定地址
    WriteCurrent(dat);        //向当前地址(上面指定的地址)写入数据
    stop();                  //停止数据传递
    delaynms(4);              //1个字节的写入周期为1ms, 最好延时1ms以上
}
/***************************************************
函数功能:从AT24Cxx中的当前地址读取数据
出口参数:x (储存读出的数据) 
***************************************************/
unsigned char ReadCurrent()
{
    unsigned char x;
    start();                   //开始数据传递
    WriteCurrent(OP_READ);   //选择要操作的AT24Cxx芯片,并告知要读其数据
    x=ReadData();            //将读取的数据存入x
    stop();                   //停止数据传递
    return x;                   //返回读取的数据
}
/***************************************************
函数功能:从AT24Cxx中的指定地址读取数据
入口参数:set_addr
出口参数:x 
***************************************************/
unsigned char ReadSet(unsigned char set_addr)
{
    start();                  //开始数据传递
    WriteCurrent(OP_WRITE); //选择要操作的AT24Cxx芯片,并告知要对其写入数据
    WriteCurrent(set_addr);    //写入指定地址
    return(ReadCurrent());   //从指定地址读出数据并返回
}
/***************************************************
函数功能:主函数
***************************************************/
main(void)
{
  unsigned char i,j;
  unsigned char temp;   //储存压缩后的音频
  unsigned char Ji;     //储存音符节拍
  unsigned char  N;       //储存音符的最大个数以在AT24C02中为音符和节拍分配存储空间
  unsigned int fr;     //储存解压缩后的音频                                
  //以下是《渴望》片头曲的一段简谱
   unsigned  int code f[]={re,mi,re,dao,l_la,dao,l_la,
                           l_sao,l_mi,l_sao,l_la,dao,
                                  l_la,dao,sao,la,mi,sao,
                                  re,                    
                                  mi,re,mi,sao,mi,
                                  l_sao,l_mi,l_sao,l_la,dao,
                           l_la,l_la,dao,l_la,l_sao,l_re,l_mi,
                                    l_sao,
                                    re,re,sao,la,sao,
                                    fa,mi,sao,mi,
                                    la,sao,mi,re,mi,l_la,dao,
                                    re,
                                    mi,re,mi,sao,mi,
                                    l_sao,l_mi,l_sao,l_la,dao,
                                    l_la,dao,re,l_la,dao,re,mi,
                                    re,
                                    l_la,dao,re,l_la,dao,re,mi,
                                    re,
                                    0x00};   //以频率0x00作为简谱的结束标志                    
//以下是简谱中每个音符的节拍
 unsigned char code JP[ ]={4,1,1,4,1,1,2,
                             2,2,2,2,8,
                                    4,2,3,1,2,2,
                                    10,
                                    4,2,2,4,4,
                                    2,2,2,2,4,
                             2,2,2,2,2,2,2,
                                    10,
                                    4,4,4,2,2,
                                    4,2,4,4,
                                    4,2,2,2,2,2,2,
                                    10,
                                    4,2,2,4,4,
                                    2,2,2,2,6,
                                    4,2,2,4,1,1,4,
                                    10,
                                    4,2,2,4,1,1,4,
                                    10
                                };
       EA=1;         //开总中断
     ET0=1;        //定时器T0中断允许
    TMOD=0x00;    // 使用定时器T0的模式1(13位计数器)
      SDA = 1;                // SDA=1,SCL=1,使主从设备处于空闲状态
    SCL = 1;             
    while(1)       //无限循环
        {
               i=0;   //从第1个音符频率f[0]开始写入AT24C02
              while(f[i]!=0x01)            //只要没有读到结束标志就继续写入
                 {
                     temp=(unsigned char)(f[i]/8); //将音频压缩为较小的字符变量
                 WriteSet(0x00+i,temp);       //在指定地址写入数据压缩后的音频
                     i++;                         //指向下一个音符音频
                  }
                  N=i;      //将音符的最大个数存于N
                  i=0;      //从第一个音符节拍JP[0]开始写入AT24C02
               while(f[i]!=0x00)
                  {
                    WriteSet(0x00+N+i,JP[i]);  //在指定地址写入音符的节拍
                    i++;                       //指向下一个音符音频
                  }
              for(i=0;i<N;i++)    
                  {
                     temp=ReadSet(0x00+i);  //读出音频
                      Ji=ReadSet(0x00+N+i);  //读出节拍
                      fr=8*temp;             //将音频解压
                      C=460830/fr;              //定时常数的计算公式
                  TH0=(8192-C)/32;       //可证明这是13位计数器TH0高8位的赋初值方法
                  TL0=(8192-C)%32;       //可证明这是13位计数器TL0低5位的赋初值方法
                  TR0=1;                 //启动定时器T0
                     for(j=0;j<Ji;j++)      //控制节拍数
                      delay();           //延时1个节拍单位
                      TR0=0;                    //关闭定时器T0                                
                  }  
                  sound=1;                  //播放完毕后,关闭蜂鸣器
               for(i=0;i<8;i++)            //播放完毕后,停顿一段时间后继续播放
                    delay();    
        }             
}


/
/***********************************************************
函数功能:定时器T0的中断服务子程序,使P3.7引脚输出音频的方波
************************************************************/
  void Time0(void ) interrupt 1 using 1  
  {
         
       TH0=(8192-C)/32;   //可证明这是13位计数器TH0高8位的赋初值方法
       TL0=(8192-C)%32;   //可证明这是13位计数器TL0低5位的赋初值方法    
        sound=!sound;     //将P3.7引脚输出电平取反,形成方波
  }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/311061.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Winclone Pro 10 for Mac:轻松备份和还原你的Windows系统

Winclone Pro 10 for Mac是一款专为Mac用户设计的备份和还原软件&#xff0c;旨在帮助用户轻松管理和保护他们的Windows系统。无论是为了数据安全还是系统的稳定性&#xff0c;Winclone Pro 10都能提供全面的解决方案。 这款软件具备强大的备份功能&#xff0c;能够快速而准确…

【Transformer】深入理解Transformer模型2——深入认识理解(下)

前言 Transformer模型出自论文&#xff1a;《Attention is All You Need》 2017年 近年来&#xff0c;在自然语言处理领域和图像处理领域&#xff0c;Transformer模型都受到了极为广泛的关注&#xff0c;很多模型中都用到了Transformer或者是Transformer模型的变体&#xff0…

【LeetCode每日一题】1185.一周中的第几天(模拟+调用库函数:三种方法)

2023-12-30 文章目录 一周中的第几天方法一&#xff1a;模拟思路步骤 方法二&#xff1a;调用库函数方法三&#xff1a;调用库函数 一周中的第几天 ​ 提示&#xff1a;给出的日期一定是在 1971 到 2100 年之间的有效日期。 方法一&#xff1a;模拟 思路 1.可以根据1970年的…

多输入多输出 | MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多输入多输出预测

多输入多输出 | MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多输入多输出预测 目录 多输入多输出 | MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多输入多输出预测预测效果基本介绍模型背景程序设计参考资料 预测效果 基本介绍 MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多输入…

1.2.0 IGP高级特性之FRR

理论部分参考文档&#xff1a;Segment Routing TI-LFA FRR保护技术 - 华为 一、快速重路由技术 FRR(Fast Reroute)快速重路由 实现备份链路的快速切换&#xff0c;也可以与BFD联动实现对故障的快速感知。 随着网络的不断发展&#xff0c;VoIP和在线视频等业务对实时性的要求越…

海康visionmaster-Group 循环:获取 Group 循环数 据结果的方法

如何获取 Group 循环的所有数据结果&#xff1f; 解答 在 Group 中使用数据集合模块&#xff0c;然后在 Group 的输出设置订阅数据集合模块相关结果&#xff0c;最后在二次开发中获取 Group 的数据结果。如下图所示&#xff0c;参数 out1 为 Group 订阅的数据集合&#xff0c;在…

Vue2中使用echarts,并从后端获取数据同步

一、安装echarts npm install echarts -S 二、导入echarts 在script中导入&#xff0c;比如&#xff1a; import * as echarts from "echarts"; 三、查找要用的示例 比如柱状图 四、初始化并挂载 <template><div id"total-orders-chart" s…

07-C++ 异常

异常 1. 概念 异常事件&#xff08;如&#xff1a;除 0 溢出&#xff0c;数组下标越界&#xff0c;所要读取的文件不存在,空指针&#xff0c;内存不足等等&#xff09; 在C 语言对错误的处理是两种方法&#xff1a; 一是使用整型的 返回值标识错误&#xff1a;二是使用 errno…

分享一个学习Typescript最全的Github网站

一个专注研究Typescript的网站&#xff0c;&#x1f396;&#x1f396;&#x1f396;在这里你可以全面深入学习Typescript相关知识,通过动画方式讲解TS&#xff0c;还有很多常见问题解答。你还可以挑战相应的题目&#xff0c;快来学习吧 我就懒一点&#xff0c;直接原滋原味的…

Django学习3——靓号管理

目录 靓号管理 表结构和数据 根据表结构的需求&#xff0c;在models.py中创建类&#xff08;由类生成数据库中的表&#xff09; 在数据库生成表 自己在数据模拟创建一些数据&#xff1a; 靓号列表 新建靓号 编辑靓号 删除靓号 搜索靓号 靓号管理 表结构和数据 根…

Dirichlet Process (徐亦达老师)狄利克雷过程

混合高斯模型的例子 混合高斯模型 混合高斯模型&#xff08;Mixture of Gaussians&#xff0c;简称GMM&#xff09;是一种概率模型&#xff0c;用于对复杂的数据分布进行建模。它是由多个高斯分布组合而成的混合模型&#xff0c;每个高斯分布&#xff08;称为组件&#xff09;…

安全加固指南:如何更改 SSH 服务器的默认端口号

在 Linux 系统中修改 SSH 服务的默认端口号是一项重要的安全措施&#xff0c;它可以帮助增强系统的安全性。这个过程相对简单&#xff0c;但必须由具有管理员权限的用户来执行。下面&#xff0c;我将向大家介绍如何安全地更改 SSH 端口的具体步骤。 1 备份 SSH 配置文件 在修改…