UART接続をさらに追求するために、また、手元にあったGrove – Serial Camera Kitを、M5STICK-CのGROVE端子に接続して、写真を取得してみようと思った。
一般の方は、撮影した写真をM5STICK-Cの液晶に表示するだろうけど、M5STICK-Cのライブラリ ver0.05は、M5STACKのように、JPGデータを表示するAPIがまだ実装されていない(コメントアウトされている)。
M5STACKみたいに、TFカードスロットが搭載されていないので、メディアにも書き出しできない。
じゃあ、汎用I/Oを使ってSPIインターフェースを使ってTFカードスロットを搭載するという手段も考えられるけど、3つしかGPIOがないので、実装できない。
ここで、諦める訳にいかない私は、文字列で画像を表示する手段に打って出ることにした。
Chromeブラウザを使っている人は、下記の文字列(長いけど)をURLにコピペしてみましょう!

すると、どうでしょう!画像が表示されませんでしたか?
HTMLファイルの中に画像を埋め込んで表示させる方法がありまして、これを利用しています。詳細は、リンクの記事読んでね。
ソースコード
// M5STICK-CとGrove - Serial Camera Kiを繋いで、文字列(Base64)で、写真を取得するプログラム
// Programed by Kazuyuki Eguchi
#include <M5StickC.h>
#include <base64.h>
#define PIC_MAX_SIZE 10000
#define PIC_PKT_LEN 128 //data length of each read, dont set this too big because ram is limited
#define PIC_FMT_VGA 7
#define PIC_FMT_CIF 5
#define PIC_FMT_OCIF 3
#define CAM_ADDR 0
#define PIC_FMT PIC_FMT_CIF
HardwareSerial CAM_SERIAL(1);
const byte cameraAddr = (CAM_ADDR << 5); // addr
unsigned long picTotalLen = 0; // picture length
unsigned char image_data[PIC_MAX_SIZE];
unsigned long image_length = 0;
void setup() {
M5.begin();
CAM_SERIAL.begin(9600, SERIAL_8N1, 33, 32); // GROVE端子の場合
initialize();
preCapture();
}
void loop() {
Capture();
GetData();
}
void clearRxBuf()
{
while (CAM_SERIAL.available())
{
CAM_SERIAL.read();
}
}
void sendCmd(char cmd[], int cmd_len)
{
for (char i = 0; i < cmd_len; i++) CAM_SERIAL.write(cmd[i]);
}
int readBytes(char *dest, int len, unsigned int timeout)
{
int read_len = 0;
unsigned long t = millis();
while (read_len < len)
{
while (CAM_SERIAL.available()<1)
{
if ((millis() - t) > timeout)
{
return read_len;
}
}
*(dest+read_len) = CAM_SERIAL.read();
// Serial.write(*(dest+read_len));
read_len++;
}
return read_len;
}
void initialize()
{
char cmd[] = {0xaa,0x0d|cameraAddr,0x00,0x00,0x00,0x00} ;
unsigned char resp[6];
Serial.print("initializing camera...");
while (1)
{
sendCmd(cmd,6);
if (readBytes((char *)resp, 6,1000) != 6)
{
Serial.print(".");
continue;
}
if (resp[0] == 0xaa && resp[1] == (0x0e | cameraAddr) && resp[2] == 0x0d && resp[4] == 0 && resp[5] == 0)
{
if (readBytes((char *)resp, 6, 500) != 6) continue;
if (resp[0] == 0xaa && resp[1] == (0x0d | cameraAddr) && resp[2] == 0 && resp[3] == 0 && resp[4] == 0 && resp[5] == 0) break;
}
}
cmd[1] = 0x0e | cameraAddr;
cmd[2] = 0x0d;
sendCmd(cmd, 6);
//Serial.println("\nCamera initialization done.");
}
void preCapture()
{
char cmd[] = { 0xaa, 0x01 | cameraAddr, 0x00, 0x07, 0x00, PIC_FMT };
unsigned char resp[6];
while (1)
{
clearRxBuf();
sendCmd(cmd, 6);
if (readBytes((char *)resp, 6, 100) != 6) continue;
if (resp[0] == 0xaa && resp[1] == (0x0e | cameraAddr) && resp[2] == 0x01 && resp[4] == 0 && resp[5] == 0) break;
}
}
void Capture()
{
char cmd[] = { 0xaa, 0x06 | cameraAddr, 0x08, PIC_PKT_LEN & 0xff, (PIC_PKT_LEN>>8) & 0xff ,0};
unsigned char resp[6];
while (1)
{
clearRxBuf();
sendCmd(cmd, 6);
if (readBytes((char *)resp, 6, 100) != 6) continue;
if (resp[0] == 0xaa && resp[1] == (0x0e | cameraAddr) && resp[2] == 0x06 && resp[4] == 0 && resp[5] == 0) break;
}
cmd[1] = 0x05 | cameraAddr;
cmd[2] = 0;
cmd[3] = 0;
cmd[4] = 0;
cmd[5] = 0;
while (1)
{
clearRxBuf();
sendCmd(cmd, 6);
if (readBytes((char *)resp, 6, 100) != 6) continue;
if (resp[0] == 0xaa && resp[1] == (0x0e | cameraAddr) && resp[2] == 0x05 && resp[4] == 0 && resp[5] == 0) break;
}
cmd[1] = 0x04 | cameraAddr;
cmd[2] = 0x1;
while (1)
{
clearRxBuf();
sendCmd(cmd, 6);
if (readBytes((char *)resp, 6, 100) != 6) continue;
if (resp[0] == 0xaa && resp[1] == (0x0e | cameraAddr) && resp[2] == 0x04 && resp[4] == 0 && resp[5] == 0)
{
if (readBytes((char *)resp, 6, 1000) != 6)
{
continue;
}
if (resp[0] == 0xaa && resp[1] == (0x0a | cameraAddr) && resp[2] == 0x01)
{
picTotalLen = (resp[3]) | (resp[4] << 8) | (resp[5] << 16);
Serial.print("picTotalLen:");
Serial.println(picTotalLen);
break;
}
}
}
}
void GetData()
{
unsigned int pktCnt = (picTotalLen) / (PIC_PKT_LEN - 6);
if ((picTotalLen % (PIC_PKT_LEN-6)) != 0) pktCnt += 1;
char cmd[] = { 0xaa, 0x0e | cameraAddr, 0x00, 0x00, 0x00, 0x00 };
unsigned char pkt[PIC_PKT_LEN];
image_length = 0;
for (unsigned int i = 0; i < pktCnt; i++)
{
cmd[4] = i & 0xff;
cmd[5] = (i >> 8) & 0xff;
int retry_cnt = 0;
retry:
delay(10);
clearRxBuf();
sendCmd(cmd, 6);
uint16_t cnt = readBytes((char *)pkt, PIC_PKT_LEN, 200);
unsigned char sum = 0;
for (int y = 0; y < cnt - 2; y++)
{
sum += pkt[y];
}
if (sum != pkt[cnt-2])
{
if (++retry_cnt < 100) goto retry;
else break;
}
if(picTotalLen < PIC_MAX_SIZE)
{
int co;
for(co = 0; co < (cnt-6) ; co++)
{
image_data[image_length] = pkt[co + 4];
image_length++;
}
}
}
if(picTotalLen < PIC_MAX_SIZE)
{
Serial.print("data:image/jpeg;base64,");
String encoded = base64::encode(image_data, image_length);
Serial.println(encoded);
}
cmd[4] = 0xf0;
cmd[5] = 0xf0;
sendCmd(cmd, 6);
}
動作結果(例)デバッグログ上
以上、ご参考までに