MIDI码深度解析

10 篇文章 2 订阅

MIDI 协议即数字音乐接口(Musical Instrument Digital Interface),是电子乐器、合成器等演奏设备之间的一种即时通信协议,用于硬件之间的实时演奏数据传递。如果理解还不够深刻,官方如下解释:

请添加图片描述

常用midi硬件接口如5芯插头串口midi和USB midi设备(usb midi任意一种usb口都行,只要usb枚举成midi设备即可),硬件midi串口接口如下:
请添加图片描述
好了,有了midi硬件,硬件中通信的数据,有个标准,我们叫做midi码,本章将重点介绍midi码如何组成的,解析思路,usb midi 和串口midi之间的封包差异。

相关链接:

MIDIOX

Bus Hound

USB MIDI Devices 1.0 | USB-IF

MIDI 1.0

1.midi码的基本格式

midi码分为状态码和数据码,状态码在字节最高位写1,数据码在字节最高位写0。状态码可以看出当前midi事件的具体用途,数据码当然是具体内容参数了。
请添加图片描述

如下格式,就是一个标准midi码
请添加图片描述

2.midi状态码分类

我们已经知道midi状态码字节最高位写1,那么如何区分是什么类型状态码呢?在【4:6】位用于区分状态码具体内容,注意midi 系统码,0xfn具体的内容功能众多,需要根据不同类型帧分析

//midi码事件类型
#define MIDI_STATUS_NOTE_OFF         0x80
#define MIDI_STATUS_NOTE_ON          0x90
#define MIDI_STATUS_AFTERTOUCH       0xA0
#define MIDI_STATUS_CONTROL_CHANGE   0xB0
#define MIDI_STATUS_PROGRAM_CHANGE   0xC0
#define MIDI_STATUS_CHANNEL_PRESSURE 0xD0
#define MIDI_STATUS_PITCH_WHEEL      0xE0
//#define MIDI_STATUS_SYSTEM          0xF0

//系统码具体类型
/*消息类型为系统消息时,低四位的数据定义*/
#define MIDI_SYSEX_Start                      0xF0   //系统独有的格式
#define MIDI_SYSEX_End                        0xF7

#define MIDI_SYS_MTC_Quarter_Frame_Message	  0xF1   //2
#define MIDI_SYS_Song_Position_Pointer		  0xF2   //3
#define MIDI_SYS_Song_Select				  0xF3   //2
#define MIDI_SYS_Tune_Request				  0xF6   //1                
#define MIDI_SYS_Clock						  0xF8   //1
#define MIDI_SYS_Tick						  0xF9   //1
#define MIDI_SYS_Start						  0xFA   //1
#define MIDI_SYS_Stop						  0xFC   //1
#define MIDI_SYS_Continue					  0xFB   //1
#define MIDI_SYS_ActiveSense				  0xFE   //1
#define MIDI_SYS_Reset						  0xFF   //1

3.midi 通道区分

除去系统码,midi通道在低四位中有效【0:3】
请添加图片描述
16个通道就像16个人!对各个通道的操作就是对各个人的操作,你叫他们干嘛就干嘛,你可以叫某个人唱歌、闭嘴、以钢琴的音色唱歌,以吉他的音色唱歌、声音多大、在别人唱了多久开始唱。在同一时间里可以有多个人发声,系统将同一时间所有人的声音合成一个声音再发出去。

4.midi状态码具体功能(系统码下一章介绍)

  • Note On 开音码(0x9n)

顾名思义,就是让某个音符发音,数据参数1:为值键,或者说是音符,就简单理解成触发某个音源发声;数据参数2:力度,或者说是音量大小。注:力度参数为0时,可当作关音码使用

00165716  KEY  2     90    30    14    1  C  3 Note On  
0016578C  KEY  2     80    30    40    1  C  3 Note Off 
00165B07  KEY  2     90    30    08    1  C  3 Note On  
00165B7E  KEY  2     80    30    40    1  C  3 Note Off 
00165F08  KEY  2     90    30    00    1  C  3 Note Off 
00165F7B  KEY  2     80    30    40    1  C  3 Note Off 

请添加图片描述

  • Note Off 关音码 (0x8n)

关掉某个音符,可直接理解成让某个音源停止发声。
请添加图片描述

注意:开音码和关音码需要成对出现,不能有丢包现象

  • 触后音 (0xAn)

触后音:Key Aftertouch,这个用在钢琴的特殊技法中,大概就是在某个音反复按压过程的一种技巧。数据参数1:值键;数据参数2:力度。具体解释如下官方文档:
请添加图片描述

  • 控制改变 (0xBn)

控制改变:Control Change,所以也被叫做CC码。数据参数1:控制ID;数据参数2:控制参数。CC码用处十分广泛,可以自定义,用于某些特殊功能,比如控制某个音色效果参数,音量,开关等等。

00187E4F  KEY  2     B0    40    00    1  ---  CC: Pedal (Sustain)
00187EE3  KEY  2     B0    40    7F    1  ---  CC: Pedal (Sustain)

请添加图片描述

CC码在midi标准中实际是指定了某些具体ID是用来干什么的,但是随着乐曲效果器发展,后面很多ID都被厂商自定义使用了,如下标准ID指定:
请添加图片描述请添加图片描述

  • 乐器改变 (0xCn)

乐器改变:Program Change 也叫做PC码。数据参数1:改变乐器ID。通常电子乐器中可用于音源改变(如,让电子钢琴发吉他贝斯音色等等),在效果器中可以用于预设切换功能。

00177A4D  KEY  2     C0    00    --    1  ---  PC: Acc. Grand Piano  

请添加图片描述

  • 通道压力 (0xDn)

通道压力:Channel Aftertouch Pressure ,从文章中了解,这玩意儿跟0xAn功能有点类似,处理多个Key的时候可能会调用这个。笔者看到文档有说数据两个字节的,也有说一个字节的,因为常规使用没有遇到过,大家遇到可以回复下(笔者这里比较坚持是一个字节的说法,感觉两个字节的那个Note已经没有啥意义了)
请添加图片描述
请添加图片描述

  • 弯音轮 (0xEn)

弯音轮:pitch wheel ,特殊技法,会发出特殊音色。数据参数两个字节,14bit有效,所以有效参数在0~0x3FFF

001DA6DD  MOX  2     E0    00    44    1  ---  Pitch Bend

请添加图片描述
请添加图片描述

5.midi 系统码具体功能

  • System Exclusive (0xF0 0xF7)

    系统独占码,0xF0开始,0xF7截至,0xF0后第一个字节可以表示制造商ID(也有其他功能),早期MIDI有这么定义

    请添加图片描述

    文档有备注的厂商入下:

    Sequential Circuits 1
    Big Briar 2
    Octave / Plateau 3
    Moog 4
    Passport Designs 5
    Lexicon 6
    Kurzweil 7
    Fender 8
    Gulbransen 9
    Delta Labs 0x0A
    Sound Comp. 0x0B
    General Electro 0x0C
    Techmar 0x0D
    Matthews Research 0x0E
    Oberheim 0x10
    PAIA 0x11
    Simmons 0x12
    DigiDesign 0x13
    Fairlight 0x14
    Peavey 0x1B
    JL Cooper 0x15
    Lowery 0x16
    Lin 0x17
    Emu 0x18
    Bon Tempi 0x20
    S.I.E.L. 0x21
    SyntheAxe 0x23
    Hohner 0x24
    Crumar 0x25
    Solton 0x26
    Jellinghaus Ms 0x27
    CTS 0x28
    PPG 0x29
    Elka 0x2F
    Cheetah 0x36
    Waldorf 0x3E
    Kawai 0x40
    Roland 0x41
    Korg 0x42
    Yamaha 0x43
    Casio 0x44
    Akai 0x45
    

    系统独占码还有如下Master Volume功能(其实用CC码自定义也可以):

    0xF0 SysEx
    0x7F Realtime
    0x7F The SysEx channel. Could be from 0x00 to 0x7F.Here we set it to "disregard channel".
    0x04 Sub-ID -- Device Control
    0x01 Sub-ID2 -- Master Volume
    0xLL Bits 0 to 6 of a 14-bit volume
    0xMM Bits 7 to 13 of a 14-bit volume
    0xF7 End of SysEx
    

    除此之外还有很多很多,具体查阅文档了解

  • MTC Quarter Frame Message (0xF1)
    0013E3B1  MOX  2     F1    30    --   --  ---  MTC Quarter Frame
    

请添加图片描述

  • Song Position Pointer (0xF2)
 00122F9F  MOX  2     F2    00    00   --  ---  Song Position Ptr

请添加图片描述

  • Song Select (0xF3)

请添加图片描述

  • Tune Request (0xF6)

请添加图片描述

  • Tick (0xF9)

需要 start stop continue 配合使用
请添加图片描述

  • MIDI Clock (0XF8)、MIDI Start (0xFA) 、 MIDI Stop (0xFB) 、MIDI Continue (0xFC)

这里放到一起说,因为midi的时钟同步需要这几个midi共同实现:

 TIMESTAMP IN PORT STATUS DATA1 DATA2 CHAN NOTE EVENT               
 00002689  MOX  2     FB    --    --   --  ---  Continue              
 0000268B  MOX  2     F8    --    --   --  ---  Timing Clock          
 000026A4  MOX  2     F8    --    --   --  ---  Timing Clock          
 000026BD  MOX  2     F8    --    --   --  ---  Timing Clock          
 000026D6  MOX  2     F8    --    --   --  ---  Timing Clock          
 000026EF  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002707  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002720  MOX  2     F8    --    --   --  ---  Timing Clock          
 0000273A  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002753  MOX  2     F8    --    --   --  ---  Timing Clock          
 0000276D  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002784  MOX  2     F8    --    --   --  ---  Timing Clock          
 0000279E  MOX  2     F8    --    --   --  ---  Timing Clock          
 000027B7  MOX  2     F8    --    --   --  ---  Timing Clock          
 000027D0  MOX  2     F8    --    --   --  ---  Timing Clock          
 000027E9  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002801  MOX  2     F8    --    --   --  ---  Timing Clock          
 0000281B  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002834  MOX  2     F8    --    --   --  ---  Timing Clock          
 0000284C  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002866  MOX  2     F8    --    --   --  ---  Timing Clock          
 0000287E  MOX  2     F8    --    --   --  ---  Timing Clock          
 00002898  MOX  2     F8    --    --   --  ---  Timing Clock          
 000028B1  MOX  2     F8    --    --   --  ---  Timing Clock          
 000028BE  MOX  2     FC    --    --   --  ---  Stop                  
  • Active Sense (0xFE)

请添加图片描述

  • Reset (0xFF)

请添加图片描述

6.midi 码的简发模式

大概意思就是,如果midi在第一次发送了某个事件类型,那么接下来如果重复发送,可以将事件类型省略(这里吐槽一下,当年因为走马观花没把文档读完,封装代码的时候没有考虑简写模式,导致跟某些乐器不兼容情况)。如下,省略开音码方法:
请添加图片描述

注:关于为什么要简写,应该是但当年midi设备波特率为:31250原因,带宽过小,某些设备需要发送的midi事件过多造成的

7.usb midi中封包要求(参考文档,midi10,usb官网可以下载)

usb midi是块传输设备,但在midi限制下,强制每帧格式为:4Byte,具体如下:
请添加图片描述

这里我们可以知道,midi数据大小可以通过cin码来确定:
请添加图片描述
请添加图片描述

举例如下封包:
请添加图片描述
这里我们通过midiox 和 bus hound软件,看看midi码是如何呈现的:
请添加图片描述

8.usb midi 描述符详细内容 ctrl+midi

-------------------------
Configuration Descriptor:
-------------------------
0x09	bLength
0x02	bDescriptorType
0x0065	wTotalLength   (101 bytes)
0x02	bNumInterfaces
0x01	bConfigurationValue
0x00	iConfiguration
0x80	bmAttributes   (Bus-powered Device)
0x32	bMaxPower      (100 mA)

Interface Descriptor:
------------------------------
0x09	bLength
0x04	bDescriptorType
0x00	bInterfaceNumber
0x00	bAlternateSetting
0x00	bNumEndPoints
0x01	bInterfaceClass      (Audio Device Class)
0x01	bInterfaceSubClass   (Audio Control Interface)
0x00	bInterfaceProtocol   (Audio Protocol undefined)
0x00	iInterface

AC Interface Header Descriptor:
------------------------------
0x09	bLength
0x24	bDescriptorType
0x01	bDescriptorSubtype
0x0100	bcdADC
0x0009	wTotalLength   (9 bytes)
0x01	bInCollection
0x01	baInterfaceNr(1)

Interface Descriptor:
------------------------------
0x09	bLength
0x04	bDescriptorType
0x01	bInterfaceNumber
0x00	bAlternateSetting
0x02	bNumEndPoints
0x01	bInterfaceClass      (Audio Device Class)
0x03	bInterfaceSubClass   (MIDI Streaming Interface)
0x00	bInterfaceProtocol   (Audio Protocol undefined)
0x00	iInterface

MS Interface Header Descriptor:
------------------------------
0x07	bLength
0x24	bDescriptorType
0x01	bDescriptorSubtype
0x0100	bcdMSC
0x0041	wTotalLength   (65 bytes)

MS MIDI IN Jack Descriptor:
------------------------------
0x06	bLength
0x24	bDescriptorType
0x02	bDescriptorSubtype
0x01	bJackType
0x01	bJackID
0x00	iJack

MS MIDI IN Jack Descriptor:
------------------------------
0x06	bLength
0x24	bDescriptorType
0x02	bDescriptorSubtype
0x02	bJackType
0x02	bJackID
0x00	iJack

MS MIDI OUT Jack Descriptor:
------------------------------
0x09	bLength
0x24	bDescriptorType
0x03	bDescriptorSubtype
0x01	bJackType
0x03	bJackID
0x01	bNrInputPins
0x02	baSourceID(1)
0x01	baSourcePin(1)
0x00	iJack

MS MIDI OUT Jack Descriptor:
------------------------------
0x09	bLength
0x24	bDescriptorType
0x03	bDescriptorSubtype
0x02	bJackType
0x04	bJackID
0x01	bNrInputPins
0x01	baSourceID(1)
0x01	baSourcePin(1)
0x00	iJack

Endpoint Descriptor (Audio/MIDI 1.0):
------------------------------
0x09	bLength
0x05	bDescriptorType
0x81	bEndpointAddress  (IN endpoint 1)
0x02	bmAttributes      (Transfer: Bulk / Synch: None / Usage: Data)
0x0008	wMaxPacketSize    (8 bytes)
0x00	bInterval         
0x00	bRefresh
0x00	bSynchAddress

MS Bulk Data Endpoint Descriptor:
------------------------------
0x05	bLength
0x25	bDescriptorType
0x01	bDescriptorSubtype
0x01	bNumEmbMIDIJack
0x03	baAssocJackID(1)

Endpoint Descriptor (Audio/MIDI 1.0):
------------------------------
0x09	bLength
0x05	bDescriptorType
0x02	bEndpointAddress  (OUT endpoint 2)
0x02	bmAttributes      (Transfer: Bulk / Synch: None / Usage: Data)
0x0008	wMaxPacketSize    (8 bytes)
0x00	bInterval         
0x00	bRefresh
0x00	bSynchAddress

MS Bulk Data Endpoint Descriptor:
------------------------------
0x05	bLength
0x25	bDescriptorType
0x01	bDescriptorSubtype
0x01	bNumEmbMIDIJack
0x01	baAssocJackID(1)

注:在某些带midi的声卡设备,midi端点大小设置虽然64Byte或者512Byte但是最终都得按照4Byte发送

  • 18
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
根据提供的引用内容,我们可以了解到MIDI是一种数字音乐标准格式,而STM32是一种微控制器,那么STM32 MIDI文件解析就是指STM32微控制器对MIDI文件进行解析的过程。在STM32中,可以使用MIDI库来解析MIDI文件,该库提供了一些函数和数据结构,可以方便地读取MIDI文件中的音符、控制参数等信息,并将其转换为MIDI消息进行处理。在解析MIDI文件时,需要注意MIDI文件的格式,一般有标准MIDI文件和扩展MIDI文件两种格式,需要根据不同的格式进行相应的解析处理。 下面是STM32 MIDI文件解析的一些步骤: ```c // 代类型:C语言 // 读取MIDI文件头信息 void readMidiHeader(FILE *fp, MidiHeader *header); // 读取MIDI文件中的音符和控制参数等信息 void readMidiTrack(FILE *fp, MidiTrack *track); // 解析MIDI消息 void parseMidiMessage(uint8_t *message, uint8_t *status, uint8_t *data1, uint8_t *data2); // 处理MIDI消息 void handleMidiMessage(uint8_t status, uint8_t data1, uint8_t data2); // 主函数 int main(void) { // 打开MIDI文件 FILE *fp = fopen("example.mid", "rb"); if (fp == NULL) { printf("Failed to open MIDI file.\n"); return -1; } // 读取MIDI文件头信息 MidiHeader header; readMidiHeader(fp, &header); // 读取MIDI文件中的音符和控制参数等信息 MidiTrack track; readMidiTrack(fp, &track); // 解析MIDI消息并处理 uint8_t message[3]; uint8_t status, data1, data2; for (int i = 0; i < track.length; i += track.events[i].length) { memcpy(message, track.events[i].data, 3); parseMidiMessage(message, &status, &data1, &data2); handleMidiMessage(status, data1, data2); } // 关闭MIDI文件 fclose(fp); return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值