WAVE音乐格式解析(讲解加代码)

2024-03-26 04:59

本文主要是介绍WAVE音乐格式解析(讲解加代码),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前一直想用单片机的adc播放音乐,一直有这个念头,网上找了很多的代码都很复杂,就想自己写一个兼容性高的,一眼就能看懂的WAVE解析的代码。以前基本功太差,现在感觉天清了雨晴了,我又觉得我行了!(狗头)

熬了两宿,对着两幅图把之前立的flag给补上了一部分。今天真的是写的太烦了,下边付了一份自己写的代码,代码目前的状态也就是能用吧!不稳定,bug超多,file safe啥的也没做,以后闲的蛋疼的时候在慢慢改。

代码先奉上,讲解日后心情好的时候补上。

 

 

 wave.c

#include <stdio.h>
#include<stdlib.h>
#include <string.h>
#include "wave.h"void printf_c(void * data,unsigned char num)
{unsigned int i = 0;char name[num];memcpy(name,data,num);for(i=0;i<num;i++)printf("%c",name[i]);printf("\n");
}int wave_asf(wave *ral_wave)
{unsigned int read_len = 0;unsigned int read_len_sizeof = 0;unsigned int ChunkID = 0;unsigned int local_offset = 0;unsigned char * extern_data;Wave_RIFF *baseframe;Wave_Base *miniframe;Wave_FMT *wavefmt;WAVE_DATA *wavedata;baseframe = (Wave_RIFF*)calloc(sizeof(Wave_RIFF),sizeof(char));miniframe = (Wave_Base*)calloc(sizeof(Wave_Base),sizeof(char));wavefmt   = (Wave_FMT*) calloc(sizeof(Wave_FMT),sizeof(char));wavedata  = (WAVE_DATA*)calloc(sizeof(WAVE_DATA),sizeof(char));read_len = fread(baseframe, 1, sizeof(Wave_RIFF), ral_wave->fp);if(read_len<sizeof(Wave_RIFF))printf("read file false!\n");local_offset = local_offset + read_len;bit8to32(&ChunkID,baseframe->ChunkID);if(ChunkID != RIFF_id)printf("file no wave!\n");while(TURE){memset(miniframe,0,sizeof(Wave_Base));ChunkID = 0;// 获取chunk_idread_len = fread(miniframe, 1, sizeof(Wave_Base), ral_wave->fp);if(read_len<sizeof(Wave_Base))printf("read file false!\n");local_offset = local_offset + read_len;bit8to32(&ChunkID,miniframe->ChunkID);switch(ChunkID){case fmt_id:read_len = fread(&read_len_sizeof, 1,sizeof(int) , ral_wave->fp);local_offset = local_offset + read_len;read_len = fread(wavefmt, 1,read_len_sizeof , ral_wave->fp);local_offset = local_offset + read_len;ral_wave->AudioFormate  = wavefmt->AudioFormate;ral_wave->BitePerSample = wavefmt->BitePerSample;ral_wave->BlockAilgn    = wavefmt->BlockAilgn;ral_wave->ByteRate      = wavefmt->ByteRate;ral_wave->NumChannels   = wavefmt->NumChannels;ral_wave->SampleRate    = wavefmt->SampleRate;ral_wave->ChunkSize     = read_len_sizeof;break;case data_id:read_len = fread(wavedata, 1,sizeof(WAVE_DATA) , ral_wave->fp);local_offset = local_offset + read_len;ral_wave->Subchunck2Size = wavedata->Subchunck2Size;goto out_while;default:read_len = fread(&read_len_sizeof, 1,sizeof(int) , ral_wave->fp);local_offset = local_offset + read_len;extern_data = (unsigned char*)calloc(read_len_sizeof,sizeof(char));read_len = fread(extern_data, 1,read_len_sizeof , ral_wave->fp);local_offset = local_offset + read_len;// 文件指针后移free(extern_data);break;}}
out_while:free(baseframe);free(miniframe);free(wavefmt);free(wavedata);ral_wave->DataOffset = 	local_offset;printf("ral_wave->DataOffset %x\n",ral_wave->DataOffset);printf("hello world~\n");return 0;
}int channel_select(wave *ral_wave)
{switch(ral_wave->AudioFormate){case 1:return 1;break;case 2:// 不支持其他break;default:// 不支持其他break;}return 0;
}// 声道,采样率分离
void channal_sbisample(wave *ral_wave)
{unsigned int read_len;
//	printf("ral_wave->ByteRate %d\n",ral_wave->ByteRate);
//	printf("ral_wave->BitePerSample %d\n",ral_wave->BitePerSample);
//	printf("ral_wave->BlockAilgn %d\n",ral_wave->BlockAilgn);
//	printf("ral_wave->NumChannels %d\n",ral_wave->NumChannels);
//	printf("ral_wave->SampleRate %d\n",ral_wave->SampleRate);// ByteRate == SampleRate * NumChannels * BitsPerSample/8// BlockAlign == NumChannels * BitsPerSample/8if(ral_wave->NumChannels == 1){switch(ral_wave->BitePerSample){case 8:printf("channel 1,8 bit\n");break;case 16:printf("channel 1,16 bit\n");break;default:printf("chanel 1 error!\n");break;}}else if(ral_wave->NumChannels == 2){switch(ral_wave->BitePerSample){case 8:printf("channel 2,8 bit\n");break;case 16:printf("channel 2,16 bit\n");c2b16 loca_c2b16;read_len = fread(&loca_c2b16, 1,sizeof(c2b16) , ral_wave->fp);printf("%d %d  read_len %d\n",sizeof(c2b16),sizeof(short),read_len);printf("loca_c2b16.L_c2b16 %x,loca_c2b16.R_c2b16 %x\n",loca_c2b16.L_c2b16,loca_c2b16.R_c2b16);ral_wave->DataOffset = ral_wave->DataOffset + read_len;printf("ral_wave->DataOffset %x\n",ral_wave->DataOffset);// 直接写道到DAC?break;default:printf("chanel 2 error!\n");break;}}else{printf("error!\n");}
}int main(void)
{wave *do_wave;do_wave = (wave*)calloc(sizeof(wave),sizeof(char));do_wave->fp = fopen("D:\\wave\\test.wav", "r");wave_asf(do_wave);if(channel_select(do_wave)!=1)return 0;for(int i = 0;i<1000;i++)channal_sbisample(do_wave);fclose(do_wave->fp);free(do_wave);return 0;
}

 wave.h

/** wave.h**  Created on: 2021年7月22日*      Author: uida0349*/#ifndef WAVE_H_
#define WAVE_H_#define TURE  1
#define FALSE 0// 将四个8bit的拼接成32bit的
#define bit8to32(a,b)    {*a = b[3];*a = ((*a)<<8)|b[2];*a = ((*a)<<8)|b[1];*a = ((*a)<<8)|b[0];}/*** 此处用于存储WAV的id信息* */
enum WAV_ID{RIFF_id = 0x46464952,fmt_id  = 0x20746d66,data_id = 0x61746164,
};typedef struct wave_base{//char           ExtentsID[4];unsigned char  ChunkID[4];
}Wave_Base;
typedef struct wave_riff{//char           ExtentsID[4];unsigned char  ChunkID[4];unsigned int   ChunkSize;unsigned char  Format[4];
}Wave_RIFF;
typedef struct wave_fmt{// fmt Bolckunsigned short AudioFormate;unsigned short NumChannels;unsigned int   SampleRate;unsigned int   ByteRate;unsigned short BlockAilgn;unsigned short BitePerSample;
}Wave_FMT;
typedef struct wave_data{// data Blockunsigned int   Subchunck2Size;
}WAVE_DATA;
typedef struct wave_Extents{char           ExtentsID[4];unsigned int   ExtentsSize;unsigned int   ExtentsDataLength;void *         ExtentsData;
}Extents_data;typedef struct wave_demo{FILE           *fp;// RIFF Bolckchar           ChunkID[4];unsigned int   ChunkSize;char           Format[4];// fmt Bolckchar           Subchunk1ID[4];unsigned int   Subchunk1Size;unsigned short AudioFormate;unsigned short NumChannels;unsigned int   SampleRate;unsigned int   ByteRate;unsigned short BlockAilgn;unsigned short BitePerSample;// extents BlockExtents_data   *ExtentsFrames;// data Blockchar           Subchunck2ID[4];unsigned int   Subchunck2Size;// data offsetunsigned int   DataOffset;
}wave;// 存储单声道8bit的数据
typedef unsigned char c1b8;
// 存储单声道16bit的数据
typedef unsigned short c1b16;
// 存储双声道8bit的数据
typedef struct {unsigned char L_c2b8;unsigned char R_c2b8;
}c2b8;// 存储双声道16bit的数据
typedef struct {unsigned short L_c2b16;unsigned short R_c2b16;
}c2b16;
#endif /* WAVE_H_ */

下一步的计划,一直到单片机里边,再加个timer,调度啥的,争取早点实现单片机片上DAC播放WAVE的想法。

这篇关于WAVE音乐格式解析(讲解加代码)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/847356

相关文章

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

深度解析Java @Serial 注解及常见错误案例

《深度解析Java@Serial注解及常见错误案例》Java14引入@Serial注解,用于编译时校验序列化成员,替代传统方式解决运行时错误,适用于Serializable类的方法/字段,需注意签... 目录Java @Serial 注解深度解析1. 注解本质2. 核心作用(1) 主要用途(2) 适用位置3

Java MCP 的鉴权深度解析

《JavaMCP的鉴权深度解析》文章介绍JavaMCP鉴权的实现方式,指出客户端可通过queryString、header或env传递鉴权信息,服务器端支持工具单独鉴权、过滤器集中鉴权及启动时鉴权... 目录一、MCP Client 侧(负责传递,比较简单)(1)常见的 mcpServers json 配置

Redis实现高效内存管理的示例代码

《Redis实现高效内存管理的示例代码》Redis内存管理是其核心功能之一,为了高效地利用内存,Redis采用了多种技术和策略,如优化的数据结构、内存分配策略、内存回收、数据压缩等,下面就来详细的介绍... 目录1. 内存分配策略jemalloc 的使用2. 数据压缩和编码ziplist示例代码3. 优化的

从原理到实战解析Java Stream 的并行流性能优化

《从原理到实战解析JavaStream的并行流性能优化》本文给大家介绍JavaStream的并行流性能优化:从原理到实战的全攻略,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的... 目录一、并行流的核心原理与适用场景二、性能优化的核心策略1. 合理设置并行度:打破默认阈值2. 避免装箱

Maven中生命周期深度解析与实战指南

《Maven中生命周期深度解析与实战指南》这篇文章主要为大家详细介绍了Maven生命周期实战指南,包含核心概念、阶段详解、SpringBoot特化场景及企业级实践建议,希望对大家有一定的帮助... 目录一、Maven 生命周期哲学二、default生命周期核心阶段详解(高频使用)三、clean生命周期核心阶

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

Python从Word文档中提取图片并生成PPT的操作代码

《Python从Word文档中提取图片并生成PPT的操作代码》在日常办公场景中,我们经常需要从Word文档中提取图片,并将这些图片整理到PowerPoint幻灯片中,手动完成这一任务既耗时又容易出错,... 目录引言背景与需求解决方案概述代码解析代码核心逻辑说明总结引言在日常办公场景中,我们经常需要从 W

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Java Scanner类解析与实战教程

《JavaScanner类解析与实战教程》JavaScanner类(java.util包)是文本输入解析工具,支持基本类型和字符串读取,基于Readable接口与正则分隔符实现,适用于控制台、文件输... 目录一、核心设计与工作原理1.底层依赖2.解析机制A.核心逻辑基于分隔符(delimiter)和模式匹