music_demo.lua 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. --[[
  2. 接线要求:
  3. SPI 使用常规4线解法
  4. Air105开发板 TF模块
  5. PC9 CS
  6. PB12(SPI0_CLK) CLK
  7. PB14(SPI0_MISO) MOSI
  8. PB15(SPI0_MISO) MISO
  9. 5V VCC
  10. GND GND
  11. ]]
  12. audio.on(0, function(id, event_id)
  13. if event_id == audio.MORE_DATA then
  14. sys.publish("moredata")
  15. else
  16. sys.publish("playover")
  17. end
  18. end)
  19. -- music文件夾,可以換成你自己的目录,不用加/sd/,demo里固定用TF卡目录,如果要本地目录,自行修改
  20. local music_dir = "/music/"
  21. function music_demo_start()
  22. sys.wait(1000) -- 启动延时
  23. local spiId = 0
  24. local result = spi.setup(
  25. spiId,--串口id
  26. 255, -- 不使用默认CS脚
  27. 0,--CPHA
  28. 0,--CPOL
  29. 8,--数据宽度
  30. 400*1000 -- 初始化时使用较低的频率
  31. )
  32. local TF_CS = pin.PB13
  33. gpio.setup(TF_CS, 1)
  34. -- fatfs.debug(1) -- 若挂载失败,可以尝试打开调试信息,查找原因
  35. fatfs.mount("SD", 0, TF_CS, 12000000)
  36. local data, err = fatfs.getfree("SD")
  37. -- usbapp.udisk_attach_sdhc(0)
  38. -- usbapp.start(0)
  39. -- sys.wait(600000)
  40. local buff = zbuff.create(1024)
  41. local in_buff = zbuff.create(8 * 1024 + 512)
  42. local play_list = {}
  43. if data then
  44. log.info("fatfs", "getfree", json.encode(data))
  45. local dir_nums, dir_info = fatfs.lsdir(music_dir)
  46. for k,v in pairs(dir_info) do
  47. log.info(v.size, k)
  48. play_list[v.size] = "/sd".. music_dir .. k
  49. end
  50. while true do
  51. for k,v in pairs(play_list) do
  52. log.info(v)
  53. if v:find(".mp3") or v:find(".MP3") then
  54. f = io.open(v, "rb")
  55. if f then
  56. -- log.debug("find mp3")
  57. in_buff:del()
  58. data = f:read(10)
  59. if data:sub(1, 3) == 'ID3' then
  60. in_buff:copy(nil, data)
  61. log.info("jump head", in_buff:query(6, 4, true))
  62. f:seek(SEEK_SET, in_buff:query(6, 4, true))
  63. end
  64. in_buff:del()
  65. data = f:read(2048)
  66. codecr = codec.create(codec.MP3)
  67. local result, AudioFormat, NumChannels, SampleRate, BitsPerSample, is_signed = codec.get_audio_info(codecr, data)
  68. if result then
  69. log.info("mp3 info", NumChannels, SampleRate, BitsPerSample)
  70. buff:resize(65536)
  71. in_buff:copy(nil, data)
  72. result = codec.get_audio_data(codecr, in_buff, buff)
  73. while buff:used() == 0 do
  74. log.info("need more data to decode", in_buff:used())
  75. data = f:read(2048)
  76. in_buff:copy(nil, data)
  77. result = codec.get_audio_data(codecr, in_buff, buff)
  78. end
  79. log.debug("start", audio.start(0, AudioFormat, NumChannels, SampleRate, BitsPerSample, is_signed))
  80. audio.write(0, buff)
  81. --local tick1,_ = mcu.tick64()
  82. data = f:read(2048)
  83. in_buff:copy(nil, data)
  84. result = codec.get_audio_data(codecr, in_buff, buff)
  85. --local tick2,_ = mcu.tick64()
  86. --log.debug(mcu.dtick64(tick2, tick1)/_)
  87. audio.write(0, buff)
  88. data = f:read(2048)
  89. while data and #data > 0 and not decode_error do
  90. sys.waitUntil("moredata", 2000)
  91. in_buff:copy(nil, data)
  92. result = codec.get_audio_data(codecr, in_buff, buff)
  93. while buff:used() == 0 and data and #data > 0 do
  94. log.info("need more data to decode", in_buff:used())
  95. data = f:read(2048)
  96. in_buff:copy(nil, data)
  97. result = codec.get_audio_data(codecr, in_buff, buff)
  98. end
  99. audio.write(0, buff)
  100. data = f:read(2048)
  101. end
  102. sys.waitUntil("playover", 2000)
  103. f:close()
  104. audio.stop(0)
  105. else
  106. log.debug("mp3 decode fail!")
  107. f:close()
  108. end
  109. codec.release(codecr)
  110. end
  111. end
  112. if v:find(".wav") or v:find(".WAV") then
  113. data = nil
  114. f = io.open(v, "rb")
  115. if f then
  116. buff:del()
  117. buff:copy(0, f:read(12))
  118. if buff:query(0, 4) == 'RIFF' and buff:query(8, 4) == 'WAVE' then
  119. local total = buff:query(4, 4, false)
  120. buff:copy(0, f:read(8))
  121. if buff:query(0, 4) == 'fmt ' then
  122. buff:copy(0, f:read(16))
  123. buff:seek(0, zbuff.SEEK_SET)
  124. local _, AudioFormat, NumChannels, SampleRate, ByteRate, BlockAlign, BitsPerSample = buff:unpack("<HHIIHH")
  125. log.debug("find fmt info", AudioFormat, NumChannels, SampleRate, ByteRate, BlockAlign, BitsPerSample)
  126. buff:copy(0, f:read(8))
  127. if buff:query(0, 4) ~= 'data' then
  128. buff:copy(0, buff:query(4, 4, false))
  129. buff:copy(0, f:read(8))
  130. end
  131. log.debug("start", audio.start(0, AudioFormat, NumChannels, SampleRate, BitsPerSample))
  132. SampleRate = (SampleRate * BlockAlign / 8) & ~(3)
  133. log.info("size", SampleRate)
  134. data = f:read(SampleRate)
  135. audio.write(0, data)
  136. data = f:read(SampleRate)
  137. audio.write(0, data)
  138. data = f:read(SampleRate)
  139. while data and #data > 0 do
  140. sys.waitUntil("moredata", 2000)
  141. audio.write(0, data)
  142. data = f:read(SampleRate)
  143. end
  144. sys.waitUntil("playover", 2000)
  145. audio.stop(0)
  146. end
  147. else
  148. log.debug(buff:query(0, 4), buff:query(8, 4))
  149. end
  150. f:close()
  151. end
  152. end
  153. end
  154. end
  155. else
  156. log.info("fatfs", "err", err)
  157. end
  158. end