MicArray 麦克风阵列

概述

麦克风阵列由沿板的六个麦克风和一个中心的麦克风组成,阵列板上的 12 颗 LED 可以用来可视化识别声源方位,可以用来做声源定位实验。

购买链接:

产品视图

点我下载产品规格书
点我下载产品原理图
点我下载点位图

麦克风阵列介绍

  • MEMS 麦克风 :7 个 MSM261S4030H0 组成阵列
    点我下载 MSM261S4030H0 数据手册
  • 灵敏度 : -26(dB,dBFS @1kHz 1Pa)
  • 信噪比 : 57dB(20kHz bandwidth, A-weighted)
  • 尺寸 :78.1*88.8mm
  • 灯光 :12 个 SK9822 LED 组成一个环形 LED 阵列
    点我下载 SK9822 数据手册
    • 多个 LED 通过双信号线级联
    • 8 Bit(256 级) 可调颜色,5 Bit (32 级)亮度调节
  • 连接器 :支持 2*5P 2.54mm 端子和 10P 0.5mm FPC 连接器

参数

功能特点 参数
声压级 140 dB SPL
灵敏度 -26(dB,dBFS @1kHz 1Pa)
信噪比 57 dB (20kHz bandwidth,A-weighted)
THD<1% (100dB SPL @1kHz S=Nom,Rload>2k )
时钟频率 1.0-4.0Mhz(正常模式)
150-800khz(低功耗模式)

模块引脚定义

引脚序号 引脚名称 类型 引脚说明
1 VIN VCC 模块电源输入正
2 GND GND 模块电源地
3 MIC_D0 I/O 0 号麦克风和1号麦克风的 I²S 接口的串行数据输出
4 MIC_D1 I/O 2 号麦克风和3号麦克风的 I²S 接口的串行数据输出
5 MIC_D2 I/O 4 号麦克风和5号麦克风的 I²S 接口的串行数据输出
6 MIC_D3 I/O 中心麦克风的 I²S 接口的串行数据输出
7 MIC_WS I/O I²S 接口的串行数据字选择
8 MIC_CK I/O I²S 接口的串行数据时钟
9 LED_CK I/O LED 的串行数据时钟
10 LED_DA I/O LED 的的串行数据输出

例程代码

接线

杜邦线接法

麦克风阵列 开发板上的对应管脚 代码中所对应的参数
VIN 3.3v/5v ——
GND GND ——
LED_CK I/O 口 sk9822_clk
LED_DA I/O 口 sk9822_dat
MIC_D0 I/O 口 i2s_d0
MIC_D1 I/O 口 i2s_d1
MIC_D2 I/O 口 i2s_d2
MIC_D3 I/O 口 i2s_d3
MIC_WS I/O 口 i2s_ws
MIC_CK I/O 口 i2s_sclk

因为 K210 的 FPIOA 的特性, 支持每个外设随意映射到任意引脚;所以将上面的表格中的 I/O 口 与自己的板子上带有数字标识的引脚连起来即可,类似于标有 GND 这种特殊标识的就不算所谓的数字标识引脚。连接完引脚后可以查看下面代码下方关于代码中引脚的说明。

烧录固件

前往 下载站 下载固件,下载 固件说明 里面的默认固件即可使用。

Micropython 代码

from Maix import MIC_ARRAY as mic
import lcd

lcd.init()
mic.init()
#mic.init(i2s_d0=23, i2s_d1=22, i2s_d2=21, i2s_d3=20, i2s_ws=19, i2s_sclk=18, sk9822_dat=24, sk9822_clk=25)

while True:
    imga = mic.get_map()
    b = mic.get_dir(imga)
    a = mic.set_led(b,(0,0,255))
    imgb = imga.resize(160,160)
    imgc = imgb.to_rainbow(1)
    a = lcd.display(imgc)
mic.deinit()

需要自行根据自己的连接的管脚号,对应着修改mic.init(i2s_d0=23, i2s_d1=22, i2s_d2=21, i2s_d3=20, i2s_ws=19, i2s_sclk=18, sk9822_dat=24, sk9822_clk=25)里面的参数。比如如果将麦克风阵列上的 MIC_D0 与 K210 板子上面的标号为 25 的引脚相连,那么对应在代码中的参数需要修改为 i2s_d0=25,其他七个引脚也是同理。因为每个人的配置是不同的,所以使用杜邦线的话没有固定的连接说明,个人自行修改引脚参数,并且不要忘记删掉mic.init(...)前面的注释(删掉#就行)。

C 语言代码

SK9822 驱动代码

仅供参考用,来源:github ;需要分析代码的话建议复制代码到电脑本地编辑器中查看。

sipeed_sk9822.c
#include "sipeed_sk9822.h"

define LED_NUM 12

define SK9822_DAT_SET() \

{                                    \
    gpiohs->output_val.bits.b27 = 1; \
}

define SK9822_DAT_CLR() \

{                                    \
    gpiohs->output_val.bits.b27 = 0; \
}

define SK9822_CLK_SET() \

{                                    \
    gpiohs->output_val.bits.b28 = 1; \
}

define SK9822_CLK_CLR() \

{                                    \
    gpiohs->output_val.bits.b28 = 0; \
}

static void sk9822_init(void)
{
gpiohs_set_drive_mode(SK9822_DAT_GPIONUM, GPIO_DM_OUTPUT);
gpiohs_set_drive_mode(SK9822_CLK_GPIONUM, GPIO_DM_OUTPUT);
gpiohs_set_pin(SK9822_DAT_GPIONUM, 0);
gpiohs_set_pin(SK9822_CLK_GPIONUM, 0);
}

void sk9822_send_data(uint32_t data)
{
for (uint32_t mask = 0x80000000; mask > 0; mask >>= 1)
{
SK9822_CLK_CLR();
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
// usleep(1);
if (data & mask)
{
SK9822_DAT_SET();
}
else
{
SK9822_DAT_CLR();
}
SK9822_CLK_SET();
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
asm volatile("nop");
// usleep(2);
}
}

//32bit 0
void sk9822_start_frame(void)
{
sk9822_send_data(0);
}

//32bit 1
void sk9822_stop_frame(void)
{
sk9822_send_data(0xffffffff);
}

//1 1 1 1 gray | b | g | r
void sk9822_data_one_led(uint8_t gray, uint8_t r, uint8_t g, uint8_t b)
{
uint32_t tosend;

gray &= 0x1f; //for make sure no error data in
tosend = ((0xe0 | gray) << 24) | (b << 16) | (g << 8) | r;
sk9822_send_data(tosend);

}

uint32_t sk9822_gen_data_one_led(uint8_t gray, uint8_t r, uint8_t g, uint8_t b)
{
uint32_t tosend;

gray &= 0x1f; //for make sure no error data in
tosend = ((0xe0 | gray) << 24) | (b << 16) | (g << 8) | r;
return tosend;

}

//first color1, then msleep interval, then color2, last msleep interval
void sk9822_flash(uint32_t color1, uint32_t color2, uint32_t interval)
{
uint8_t index;

color1 |= 0xe0000000;
color2 |= 0xe0000000;

sk9822_start_frame();
for (index = 0; index < LED_NUM; index++)
{
    sk9822_send_data(color1);
}
sk9822_stop_frame();
msleep(interval);

sk9822_start_frame();
for (index = 0; index < LED_NUM; index++)
{
    sk9822_send_data(color2);
}
sk9822_stop_frame();
msleep(interval);

}

static void arraymove(uint32_t array[], uint8_t len)
{
uint8_t index;
uint32_t tmp;

tmp = array[0];
for (index = 0; index < len - 1; index++)
{
    array[index] = array[index + 1];
}
array[len - 1] = tmp;

}
//呼吸
//跑马灯
//方位
void sk9822_horse_race(uint8_t r, uint8_t g, uint8_t b, uint32_t interval, uint8_t times)
{
uint32_t led_frame[LED_NUM] = {0};
uint8_t i, index;

for (index = 0; index < 12; index++)
{
    led_frame[index] = 0xff000000;
}

for (index = 0; index < 6; index++)
{
    // led_frame[index] = sk9822_gen_data_one_led((0xe0|(index*4)),r-40*index,g-30*index,b-20*index);
    led_frame[index] = sk9822_gen_data_one_led((0xe0 | (32 - index * 4)), r, g, b);
}

for (index = 0; index < times; index++)
{
    while (1)
    {
        sk9822_start_frame();
        for (i = 0; i < 12; i++)
        {
            sk9822_send_data(led_frame[i]);
        }
        sk9822_stop_frame();
        arraymove(led_frame, LED_NUM);
        msleep(interval);
    }
}

}

void sk9822_breath(uint8_t r, uint8_t g, uint8_t b, uint32_t interval)
{
uint8_t index, cnt, dir;

uint32_t color = sk9822_gen_data_one_led(0xff, r, g, b);

cnt = 0;
dir = 1;

while (1)
{
    if (cnt >= 30)
    {
        dir = !dir;
        cnt = 0;
    }
    cnt++;
    color = sk9822_gen_data_one_led((0xe0 | (dir ? cnt : 31 - cnt)), r, g, b);
    sk9822_start_frame();
    for (index = 0; index < LED_NUM; index++)
    {
        sk9822_send_data(color);
    }
    sk9822_stop_frame();
    msleep(interval);
}

}

void sipeed_init_mic_array_led(void)
{
sk9822_init();
//flash 3 times
sk9822_flash(0xffeec900, 0xffff0000, 200);
sk9822_flash(0xffeec900, 0xff00ff00, 200);
sk9822_flash(0xffeec900, 0xff0000ff, 200);
}

// void sipeed_calc_voice_strength(uint8_t voice_data[])
// {
// uint32_t tmp_sum[12] = {0};
// uint32_t led_color[12];
// uint8_t i, index, tmp;

// for (index = 0; index < 12; index++)
// {
// tmp_sum[index] = 0;
// for (i = 0; i < voice_strength_len[index]; i++)
// {
// tmp_sum[index] += voice_data[voice_strength[index][i]];
// }
// tmp = (uint8_t)tmp_sum[index] / voice_strength_len[index];
// led_brightness[index] = tmp > 15 ? 15 : tmp;
// }
// sk9822_start_frame();
// for (index = 0; index < 12; index++)
// {
// led_color[index] = (led_brightness[index] / 2) > 1 ? (((0xe0 | (led_brightness[index] * 2)) << 24) | 0xcd3333) : 0xe0000000;
// sk9822_send_data(led_color[index]);
// }
// sk9822_stop_frame();
// }

Micarray 代码

仅供参考用,来源:github;需要分析代码的话建议复制代码到电脑本地编辑器中查看。

其他资料

麦克风阵列硬件资料

更多使用方法

请移步到论坛上查看

常见问题

出现报错信息

报错信息

更换其他IO管脚即可