HOME | Raspberry Pi | ビジネス書籍紹介 | 2026-01-04 (Sun) Today's Access : 317 Total : 1261015. Since 10 Sep. 2019

シリアルカメラ
2024.08.24/2025.12.18更新

YouTube でも紹介しています。画像をクリックすると再生できます。

今回は、シリアルカメラを安く入手したので遊んでみました。

●SeeedStudio GROVE - シリアルカメラキット
入力電圧:5V/ 画素数:30万画素/ 解像度:640x480, 320x240, 160x120/ UART:9600~115200[bps]/ 通信方式:RS485, RS232/ 寸法:76.5x153x12[mm]/ JPEG圧縮:高/中/低品質の三種類/ AGC機能搭載/ 自動ホワイトバランス/ 自動露出制御/ フォーカス制御可能
Grove - Serial Camera Kit
Grove - シリアル カメラ キットには、2 つの仕様のレンズが同梱されています。 標準レンズは一般的な写真撮影用で、広角レンズは監視プロジェクトに特に適しています。

補足:Grove UART
M5Stack FIREやM5GOに用意されているインターフェースです。非同期のシリアル通信を行うセンサ、I/Oデバイスとのインターフェースです。

M5StackがサポートするGrove System

●CJ - CAM User Manual - OV528 Protocol
ユーザ・マニュアル記載のプロトコルをもとに実際の流れをみていきます。
cj-ov528_protocol.pdf

このシリアルカメラには11種類の命令があります。

まず最初、ホストのマイコンからカメラに向かって同期命令を送信すると、無効なコードが返ってきます。
そこで再度、同期命令を送信すると、確認応答が返ってきます。 これでセッションが確立します。

通信速度設定

通信速度を指定します。ここでは115200bpsにしています

画像形式・解像度設定

JPEG形式、解像度320*240を指定しています

Set Package Size
カメラモジュールから一度に読込む画像ブロックサイズを指定します

ブロックサイズは下位バイト、上位バイトの順に設定します。

Snapshot

JPEG画像の圧縮・非圧縮を指定します。

Get Picture
画像の取得開始を指示します


この後に画像データが続きます。 画像データの先頭6バイトはACKコードです。 画像ブロックサイズからACKコードの長さ6を引いた値で画像サイズを割り算すると、カメラ側から送信させるブロック数が求まります

ACKの5バイト目は0から始まる連番になっています。
画像ブロック数が43ならば最終連番は、43 - 1 = 0x2A です
ACKの5・6バイト目を f0:f0 とすることで画像データの終了を知らせます。

●サンプルコード

Seeed-Studio/Grove_Serial_Camera_Kit

上記サンプルコードではプッシュボタンを押したときにカメラからの画像をSDカードに保存しますが、撮影中の画像を表示する機能はありません。

●JPEGDecoder

Bodmer/JPEGDecoder
JPEGDecoderライブラリに含まれるpicojpegを使うことでJPEG画像をRGB(BMP)形式に変換することができます。 このRGBデータを用いて、TFTディスプレイに表示します。
ただし、Arduino UNOなどではRAMが少なく、このライブラリを実行することができません。
ここでは、srcディレクトリにある3つのコードを利用しました
・picojpeg.c
・picojpeg.h
・User_Config.h

●Arduino Nano ESP32 Compatible
Nano ESP32は8MBのPSRAMを実装しているので、picojpegを余裕で実行でき、カメラがらの画像をメモリ中に保持することも可能です。

Arduino Nano ESP32 には、下記のような機能があります
・豊富なメモリ(384KB ROM,512KB RAM,16MB Flash, 8MB PSRAM
・USBホスト機能
・2系統のSPI
・2系統のI2C
・3系統のシリアル
・動作電圧3.3V

ここでは「NANO ESP32 統合環境」で紹介したESP32-S3汎用機を使用します
NANO ESP32 統合環境

●ロジックレベル変換ケーブル作成
GROVE - シリアルカメラモジュールは5V駆動です。 マイコン側からの5V給電では電力不足でシリアルカメラが応答しないことがあるので、外部5V電源を用います。

4ビット双方向ロジックレベル変換モジュール BSS138使用
ESP32と接続する際にはレベル変換が必要になります。

NチャネルMOSFETを使用したロジックレベル変換モジュールです。高電圧側(HV)と低電圧側(LV)が定められています。双方向に変換できますのでI2CやSPI、UART等、データやクロックの方向を考慮する必要がありません。
電源電圧min.:2.5V / 電源電圧max.:20V

ワニ口テストクリップ
左のワニ口テストクリップのワニ口クリップ部分はカットして、電源ケーブルとして用います。
Aliexpress などから入手できます。
Grove-Stemma QT/Qwiic変換ケーブル

GROVE シリアルカメラモジュールキット添付のケーブルは両端GROVEコネクタです。 これは使わずに、Grove-Stemma QT/Qwiic変換ケーブルを使いました。
【注意】 シリアル カメラ キット基板のGROVEソケットの並びは黒(GND)黄(RX)白(TX)赤(5V)になっています。 そこでGROVE側コネクタのケーブルを黒赤白黄から黒黄白赤に差し替えています。

SHコネクターDIP化キット

Qwiicコネクタをユニバーサル基板に接続するために、Qwiic互換コネクタのブレークアウト基板を用いました。

コネクター付コード 4P 黒赤青黄

片端Qwiicコネクタの4線ケーブルです。ブレッドボード上に回路を組む際に使用しました。

基板for電池ボックス(3枚入) [P-BTB14X52-HO]

14.0x52.0mm、2.54mmピッチのユニバーサル基板です。


モバイルバッテリーからUSB接続で5Vを供給、レベル変換モジュールを組み込むとこんな感じになります。

●配線
ESP32-S3汎用機のSerial2 QWIICコネクタにカメラモジュール・ケーブルを挿します
 Camera  レベル変換  NANO ESP32 
TX(白)  --BSS138--  Serial2:TX
RX(赤)  --BSS138--  Serial2:RX
GND(黒)  --BSS138--  GND
5V(黄) --BSS138--  3.3V
 ----------  BATTERY(5V) 



●NANO ESP32ソースコード
#include <WiFi.h>
#include <Wire.h>
#include <SPI.h>
#include <SD.h>

#define  SPI_MISO      47  // D12
#define  SPI_MOSI      38  // D11
#define  SPI_SCK       48  // D13
#define  SPI_DC        21  // D10
#define  SPI_TFT_CS    18  // D9
#define  SPI_TFT_RST   17  // D8
#define  SPI_TFT_BL    10  // D7 Backlight
#define  SERIAL2_RX     9  // Serial2 D6 ->QWIIC(RX)
#define  SERIAL2_TX     8  //         D5 ->QWIIC(TX)
#define  SERIAL1_RX     6  // Serial1 D2
#define  SERIAL1_TX     5  //         D3
#define  SERIAL0_RX    44  // Serial0 D0(RX0)
#define  SERIAL0_TX    43  //         D1(TX1)
SPIClass SPI2(HSPI);
#define  SPI2_MISO      3  // microSD
#define  SPI2_MOSI      2  // 
#define  SPI2_SCK       4  // 
#define  SPI2_CS        7  // 

#define PIC_BLK_LEN    128      // data length of each read, dont set this too big because ram is limited
#define PIC_FMT_VGA    7        // 640x480
#define PIC_FMT_CIF    5        // 320x240
#define PIC_FMT_OCIF   3        // 160x128
#define CAM_SERIAL     Serial

#define SHUTTER_PIN   1
#define SDSPEED 40000000

//#define PIC_FMT        PIC_FMT_VGA
#define PIC_FMT        PIC_FMT_CIF

unsigned long picTotalLen = 0;            // picture length

#include <Adafruit_GFX.h>
#include <Adafruit_ST7789.h>
Adafruit_ST7789  tft = Adafruit_ST7789(SPI_TFT_CS, SPI_DC, SPI_MOSI, SPI_SCK, SPI_TFT_RST);

#define TFT_BLACK 0x0000
#define TFT_RED   0xF800
#define TFT_BLUE  0x001F
#define TFT_GREEN 0x07E0
#define TFT_WHITE 0xFFFF

#include "picojpeg.h"

char filenames[255];
uint8_t *jpeg = NULL;

typedef struct {
	// Header
	uint8_t Signature[2];   // 'BM'
	uint8_t FileSize[4];
	uint8_t reserved1[4];   // unused (=0)
	uint8_t DataOffset[4];
	// InfoHeader
	uint8_t Size[4];        // Size of InfoHeader =40 
	uint8_t Width[4];
	uint8_t Height[4];
	uint8_t Planes[2];      // always 1
	uint8_t BitPerPixel[2];
	uint8_t Compression[4];
	uint8_t ImageSize[4];
	uint8_t XpixelsPerM[4];
	uint8_t YpixelsPerM[4];
	uint8_t ColorsUsed[4];
	uint8_t ImportantColors[4];
} BMP_FORMAT;

BMP_FORMAT Bmp;

typedef struct {
	uint8_t  R;
	uint8_t  G;
	uint8_t  B;
} PIXEL_INFO;

PIXEL_INFO *RGBs    = NULL;
uint16_t *color565s = NULL;

typedef struct {
	uint32_t Width;
	uint32_t Height;
	uint16_t BitPerPixel;
	uint32_t FileSize;
	uint32_t DataOffset;
	uint32_t ImageSize;
	uint32_t ColorsUsed;
} IMAGE_INFO;

IMAGE_INFO img;

#define local_min(a,b)  (((a) < (b)) ? (a) : (b))

FILE *stream;
pjpeg_image_info_t image_info;
int  jpeg_is_available;
int  jpeg_mcu_x;
int  jpeg_mcu_y;
uint jpeg_read_offset;
uint jpeg_size;

// for JPEGDec - public
uint8_t *jpeg_pImage;

int jpeg_MCUSPerRow;
int jpeg_MCUSPerCol;
int jpeg_MCUWidth;
int jpeg_MCUHeight;
int jpeg_MCUx;
int jpeg_MCUy;

void clearRxBuf()
{
	while (Serial2.available()) Serial2.read();
}

void sendCmd(char cmd[], int cmd_len)
{
	for (char i = 0; i < cmd_len; i++) Serial2.write(cmd[i]);
}

void initialize()
{
	char cmd[] = {0xaa,0x0d,0x00,0x00,0x00,0x00} ; // SNYC
	char cmd_reset[] = {0xaa,0x08,0x00,0x00,0x00,0x00} ; // Reset & Reboot
	unsigned char resp[6];
	uint8_t len;

	Serial2.setTimeout(500);
	while (1) {
		//clearRxBuf();
		sendCmd(cmd,6); // SNYC

		// Receive ACK from OV528
		len = Serial2.readBytes((char *)resp, 6);
		// aa:0e:0d:00:00:00
		if (len == 6) {
			if (   resp[0] == 0xaa
					&& resp[1] == 0x0e
					&& resp[2] == 0x0d
					&& resp[4] == 0
					&& resp[5] == 0) {

				// Receive SNYC from OV528
				len = Serial2.readBytes((char *)resp, 6);
				// aa:0d:00:00:00:00
				if (len == 6) {
					if (   resp[0] == 0xaa
							&& resp[1] == 0x0d
							&& resp[2] == 0
							&& resp[3] == 0
							&& resp[4] == 0
							&& resp[5] == 0)
						break;
				}
			} else {
				// fd:3f:08:00:00:00
				sendCmd(cmd_reset,6); // Reset & Reboot
			}
		}
	}
	cmd[1] = 0x0e;  // ACK
	cmd[2] = 0x0d;
	sendCmd(cmd, 6);

	// Baud Rate 115200bps
	cmd[1] = 0x07;
	cmd[2] = 0x00;
	cmd[3] = 0x01;
	cmd[4] = 0x00;
	cmd[5] = 0x00;
	sendCmd(cmd, 6);
	len = Serial2.readBytes((char *)resp, 6);
}

void preCapture()
{
	uint8_t len;

	char cmd[] = { 0xaa, 0x01, 0x00, 0x07, 0x00, PIC_FMT };
	unsigned char resp[6];

	Serial2.setTimeout(100);
	while (1) {
		clearRxBuf();
		sendCmd(cmd, 6);
		len = Serial2.readBytes((char *)resp, 6);
		if (len == 6) {
			if (   resp[0] == 0xaa
				  && resp[1] == 0x0e
				  && resp[2] == 0x01
				  && resp[4] == 0
				  && resp[5] == 0)
				break;
		}
		initialize();
	}
}

unsigned long Capture()
{
	char cmd[] = { 0xaa, 0x06, 0x08, PIC_BLK_LEN & 0xff, (PIC_BLK_LEN>>8) & 0xff ,0};
	unsigned char resp[6];
	uint8_t len;

	picTotalLen = 0;

	Serial2.setTimeout(100);
	while (1) {
		clearRxBuf();
		sendCmd(cmd, 6);
		len = Serial2.readBytes((char *)resp, 6);
		if (len == 6) {
			if (   resp[0] == 0xaa
			    && resp[1] == 0x0e
			    && resp[2] == 0x06
			    && resp[4] == 0
			    && resp[5] == 0)
			{
				cmd[1] = 0x05;
				cmd[2] = 0;
				cmd[3] = 0;
				cmd[4] = 0;
				cmd[5] = 0;
				while (1) {
					clearRxBuf();
					sendCmd(cmd, 6);
					len = Serial2.readBytes((char *)resp, 6);
					if (len == 6) {
						if (   resp[0] == 0xaa
						    && resp[1] == 0x0e
						    && resp[2] == 0x05
						    && resp[4] == 0
						    && resp[5] == 0)
							break;
						}
					}
				}
				cmd[1] = 0x04;
				cmd[2] = 0x1;
				while (1) {
				clearRxBuf();
				sendCmd(cmd, 6);
				len = Serial2.readBytes((char *)resp, 6);
				if (len == 6) {
					if (   resp[0] == 0xaa
					    && resp[1] == 0x0e
					    && resp[2] == 0x04
					    && resp[4] == 0
					    && resp[5] == 0) {
						Serial2.setTimeout(1000);
						len = Serial2.readBytes((char *)resp, 6);
						if (len == 6) {
							if (   resp[0] == 0xaa
							    && resp[1] == 0x0a
							    && resp[2] == 0x01) {
								picTotalLen = (resp[3]) | (resp[4] << 8) | (resp[5] << 16);
								return picTotalLen;
							}
						}
					}
				}
			}
		}
		initialize();
		preCapture();
	}
	return picTotalLen;
}

boolean GetData()
{
	uint8_t *imgPt = jpeg;
	uint16_t cnt = 0;
	int retry_cnt = 0;

	unsigned int pktCnt = (picTotalLen) / (PIC_BLK_LEN - 6);
	if ((picTotalLen % (PIC_BLK_LEN-6)) != 0) pktCnt += 1;

	char cmd[] = { 0xaa, 0x0e, 0x00, 0x00, 0x00, 0x00 };
	unsigned char pkt[PIC_BLK_LEN];

	Serial2.setTimeout(1000);
	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);

		retry_cnt = 0;
		while (1) {
			cnt = Serial2.readBytes((char *)pkt, PIC_BLK_LEN);
			if (cnt > 0) break;
			if (++retry_cnt > 10) return false;
		}

		retry_cnt = 0;
		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 return false;
		}

		memcpy(imgPt, &pkt[4], cnt-6);
		imgPt += (cnt-6);
	}
	cmd[4] = 0xf0;
	cmd[5] = 0xf0;
	sendCmd(cmd, 6);
	return true;
}

boolean  saveBMP(uint32_t width, uint32_t height)
{
	File  bmpFile;
	PIXEL_INFO *rgb = RGBs;
	int   x,y;
	int   from_x, from_y, to_y;
	uint32_t pos;
	static int picNameNum = 1;
	char picName[] = "/ov528/pic00.bmp";

	while (picNameNum <= 100) {
		picName[10] = picNameNum/10 + '0';
		picName[11] = picNameNum%10 + '0';
		if (!SD.exists(picName)) break;
		picNameNum++;
	}
	if (picNameNum > 100) return false;

	img.BitPerPixel = 24;
	img.ColorsUsed  = 0;
	if (width  > 0) img.Height = height;
	if (height > 0) img.Width  = width;
	img.ImageSize  = img.Width * img.Height * 3;
	img.DataOffset = sizeof(BMP_FORMAT);
	img.FileSize   = img.DataOffset + img.ImageSize;

	memset(&Bmp, 0x00, sizeof(BMP_FORMAT));
	memcpy(&Bmp.Signature,(uint8_t*)"BM",2);
	long2byte(Bmp.FileSize,     img.FileSize);
	long2byte(Bmp.DataOffset,   img.DataOffset);
	long2byte(Bmp.Size,         40);
	long2byte(Bmp.Width,        img.Width);
	long2byte(Bmp.Height,       img.Height);
	short2byte(Bmp.Planes,      1);
	short2byte(Bmp.BitPerPixel, img.BitPerPixel);
	long2byte(Bmp.ImageSize,    img.ImageSize);
	long2byte(Bmp.ColorsUsed,   img.ColorsUsed);

	bmpFile = SD.open(picName, FILE_WRITE);
	if(!bmpFile) {
		SD.end();
		return false;
	}

	bmpFile.write((uint8_t*)&Bmp, sizeof(BMP_FORMAT));
	bmpFile.write((const uint8_t *)rgb, img.ImageSize);

	bmpFile.close();
	SD.end();

	return true;
}

static uint8_t pjpeg_need_bytes_callback(uint8_t* pBuf, uint8_t buf_size, uint8_t *pBytes_actually_read, void *pCallback_data) {
	pjpeg_callback(pBuf, buf_size, pBytes_actually_read, pCallback_data);
	return 0;
}

uint8_t pjpeg_callback(uint8_t* pBuf, uint8_t buf_size, uint8_t *pBytes_actually_read, void *pCallback_data) {
	uint n;

	n = local_min(jpeg_size - jpeg_read_offset, buf_size);

	fread(pBuf,n,1,stream);

	*pBytes_actually_read = (uint8_t)(n);
	jpeg_read_offset += n;
	return 0;
}

int JpegDec_decode(unsigned long len)
{
	jpeg_mcu_x = 0;
	jpeg_mcu_y = 0;
	jpeg_is_available = 0;
	jpeg_read_offset = 0;
	jpeg_size = len;

	uint8_t status = pjpeg_decode_init(&image_info, pjpeg_need_bytes_callback, NULL, 0);

	if (status) return -1;

	// In reduce mode output 1 pixel per 8x8 block.
	jpeg_MCUSPerRow = image_info.m_MCUSPerRow;
	jpeg_MCUSPerCol = image_info.m_MCUSPerCol;
	jpeg_MCUWidth   = image_info.m_MCUWidth;
	jpeg_MCUHeight  = image_info.m_MCUHeight;

	jpeg_pImage = new uint8_t[image_info.m_MCUWidth * image_info.m_MCUHeight * 3];
//	jpeg_pImage = (uint8_t *)malloc(jpeg_MCUWidth * jpeg_MCUHeight * 3);
	if (!jpeg_pImage) return -1;
	memset(jpeg_pImage, 0, sizeof(jpeg_pImage));

	jpeg_is_available = 1 ;

	return JpegDec_decode_mcu();
}

int JpegDec_decode_mcu(void) {

	uint8_t status = pjpeg_decode_mcu();

	if (status) {
		jpeg_is_available = 0 ;

		if (status != PJPG_NO_MORE_BLOCKS) return -1;
	}
	return 1;
}

int JpegDec_read(void)
{
	int y, x;
	uint8_t *pDst_row;

	if(jpeg_is_available == 0 || jpeg_mcu_y >= image_info.m_MCUSPerCol) return 0;

	// Copy MCU's pixel blocks into the destination RGBs.
	pDst_row = jpeg_pImage;
	for (y = 0; y < image_info.m_MCUHeight; y += 8) {
		const int by_limit = local_min(8, image_info.m_height - (jpeg_mcu_y * image_info.m_MCUHeight + y));

		for (x = 0; x < image_info.m_MCUWidth; x += 8) {
			uint8_t *pDst_block = pDst_row + x * 3;

			// Compute source byte offset of the block in the decoder's MCU buffer.
			uint src_ofs = (x * 8U) + (y * 16U);
			const uint8_t *pSrcR = image_info.m_pMCUBufR + src_ofs;
			const uint8_t *pSrcG = image_info.m_pMCUBufG + src_ofs;
			const uint8_t *pSrcB = image_info.m_pMCUBufB + src_ofs;

			const int bx_limit = local_min(8, image_info.m_width - (jpeg_mcu_x * image_info.m_MCUWidth + x));

			int bx, by;
			for (by = 0; by < by_limit; by++) {
				uint8_t *pDst = pDst_block;

				for (bx = 0; bx < bx_limit; bx++) {
					pDst[0] = *pSrcR++;
					pDst[1] = *pSrcG++;
					pDst[2] = *pSrcB++;
					pDst += 3;
				}

				pSrcR += (8 - bx_limit);
				pSrcG += (8 - bx_limit);
				pSrcB += (8 - bx_limit);
				pDst_block += jpeg_MCUWidth * 3;
			}
		}
		pDst_row += (jpeg_MCUWidth * 8 *3 );
	}

	jpeg_MCUx = jpeg_mcu_x;
	jpeg_MCUy = jpeg_mcu_y;

	jpeg_mcu_x++;
	if (jpeg_mcu_x == image_info.m_MCUSPerRow) {
		jpeg_mcu_x = 0;
		jpeg_mcu_y++;
	}

	if(JpegDec_decode_mcu() < 0) jpeg_is_available = 0 ;

	return 1;
}

void short2byte(uint8_t *pt, uint16_t val) {
	*pt     = (uint8_t)val      &0x00ff;
	*(pt+1) = (uint8_t)((val>>8)&0x00ff);
}

void long2byte(uint8_t *pt, uint32_t val) {
	*pt     = (uint8_t)val       &0x000000ff;
	*(pt+1) = (uint8_t)((val>> 8)&0x000000ff);
	*(pt+2) = (uint8_t)((val>>16)&0x000000ff);
	*(pt+3) = (uint8_t)((val>>24)&0x000000ff);
}

boolean Jpeg2RGBs(uint8_t *data, uint32_t len, uint32_t *width, uint32_t *height)
{
	uint8_t  *pImg;
	uint32_t pos;
	int      x,y,bx,by;

	if((stream = fmemopen((void*)data, len, "rb"))==NULL) return false;

	// Decoding start
	if ( JpegDec_decode(len) == -1) return false;
	*width  = image_info.m_width;
	*height = image_info.m_height;

	while(JpegDec_read()){
		pImg = jpeg_pImage;

		for(by=0; by<jpeg_MCUHeight; by++){
			for(bx=0; bx<jpeg_MCUWidth; bx++){
				x = jpeg_MCUx * jpeg_MCUWidth + bx;
				y = jpeg_MCUy * jpeg_MCUHeight + by;
				if(x<image_info.m_width && y<image_info.m_height){
					pos = x + (y * image_info.m_width);
					RGBs[pos].B = pImg[0];
					RGBs[pos].G = pImg[1];
					RGBs[pos].R = pImg[2];
					color565s[pos] = ((RGBs[pos].B >> 3)<<11) | ((RGBs[pos].G >> 2)<<5) | (RGBs[pos].R >> 3);
				}
				pImg += 3;
			}
		}
	}
	delete[] jpeg_pImage;

	return true;
}

void setup()
{
	char filename[64], *pt;
	uint32_t pos, x, y;
	uint32_t bmp_width, bmp_height;

	pinMode(SPI2_CS, OUTPUT);
	SPI2.begin(SPI2_SCK, SPI2_MISO, SPI2_MOSI, -1);
	while(!SD.begin(SPI2_CS, SPI2, SDSPEED)) delay(100);

	pinMode(SHUTTER_PIN, INPUT);    // initialize the pushbutton pin as an input
	Serial2.begin(115200, SERIAL_8N1, SERIAL2_RX, SERIAL2_TX);

	initialize();
	preCapture();

	if (!(color565s = (uint16_t*)malloc(320*240*2))) while(1);
	if (!(RGBs = (PIXEL_INFO *)malloc(320 * 240 * sizeof(PIXEL_INFO)))) while(1);

  tft.init(240, 320);  // Init ST7789
	tft.setRotation(1);

	while(1) {
		if (Capture() > 0) {
			if (jpeg) free(jpeg);
			if ((jpeg = (uint8_t*)malloc(picTotalLen))) {
				if (GetData()) {
					Jpeg2RGBs(jpeg, picTotalLen, &bmp_width, &bmp_height);
					tft.drawRGBBitmap(0, 0, color565s, bmp_width, bmp_height);
					if (digitalRead(SHUTTER_PIN)) {
						tft.setCursor(10, 10);
						tft.setTextColor(TFT_RED, TFT_BLACK);
						tft.setTextSize(1);
						tft.println("[REC]");
						saveBMP(bmp_width, bmp_height);
					}
				}
			}
		}
	}
}

void loop() {}

●撮影


■参考文献
Groveのシリアルカメラ「OV528」を調べてみる #Arduino - Qiita
GroveのシリアルカメラとOV528の違いが気になる
Raspberry Pi(ラズベリー パイ)は、ARMプロセッサを搭載したシングルボードコンピュータ。イギリスのラズベリーパイ財団によって開発されている。
2020.05.28 画像処理 第1回トイカメラ
2020.06.09 画像処理 第2回カメラモジュール制御
2020.06.28 画像処理 第3回リアルタイムクロック
2020.07.08 画像処理 第4回電源回路
2020.10.27 画像処理 第5回自作デジカメ初号機完成
2020.11.10 画像処理 第6回ドーナツデジカメ
2021.05.16 画像処理・基本変換
2021.07.10 M5Stackアプリの移植
2022.04.04 減色処理 雑談
2022.04.18 減色処理 均等量子化法とK平均法
2022.05.04 減色処理 グレースケール・二値化
2022.05.18 減色処理 二値化画像印刷
2022.08.12 京セラ feelH” Treva カメラ
2022.10.15 デジカメ弐号機 1.仕様変更
2022.10.21 デジカメ弐号機 2.SDカードとRTC
2022.11.04 デジカメ弐号機 3.SPI DISPLAY
2022.11.18 デジカメ弐号機 4.ストリーミング
2022.12.02 デジカメ弐号機 5.機能統合
2022.12.17 デジカメ弐号機 6.完成
2023.05.15 アナログ風ゲージ
2023.06.01 ADS1115デジタル電圧計
2023.10.01 立体視(ステレオグラム)
2023.10.16 漢字フォントの表示と拡張
2024.04.08 自作デジカメ参号機・雑談
2024.08.24 シリアルカメラ
2025.07.23 C3.jsによるグラフ描画
2025.09.09 東芝デジタルカメラユニット DMR-C1
2025.10.10 AQM1248A小型液晶ボード
2025.10.11 Monochrome OLED
2025.12.31 キャラクターディスプレイで遊ぶ

たいていのことは100日あれば、うまくいく。長田英知著
「時間がなくて、なかなか自分のやりたいことができない」 「一念発起して何かを始めても、いつも三日坊主で終わってしまう」 「色んなことを先延ばしにしたまま、時間だけが過ぎていく」 そこで本書では、そんな著者が独自に開発した、 まったく新しい目標達成メソッド「100日デザイン」について、 その知識と技術を、余すところなくご紹介します。

まんがで納得ナポレオン・ヒル 思考は現実化する
OLとして雑務をこなす日々に飽き足らず、科学者だった父が残した薬品を商品化すべく、起業を決意した内山麻由(27)。彼女はセミナーで知り合った謎の女性からサポートを得ながら、彼女と二人三脚でナポレオン・ヒルの成功哲学を実践し、さまざまな問題を乗り越えていく。 ヒル博士の<ゴールデンルール>に従い、仕事に、恋に全力疾走する彼女の、成功への物語。

今日は人生最悪で最高の日 1秒で世界を変えるたったひとつの方法 ひすいこたろう著
偉人の伝記を読むと、最悪な日は、不幸な日ではなく、新しい自分が始まる日であることがわかります。最悪な出来事は、自分の人生が、想像を超えて面白くなる兆しなのです。偉人伝を読むことで、このときの不幸があったおかげで、未来にこういう幸せがくるのかと、人生を俯瞰する視線が立ち上がるのです。

ご飯は私を裏切らない heisoku著
辛い現実から目を背けて食べるご飯は、いつも美味しく幸せを届けてくれる。 29歳、中卒、恋人いない歴イコール年齢。バイト以外の職歴もなく、短期バイトを転々とする日々。ぐるぐると思索に耽るけど、ご飯を食べると幸せになれる。奇才の新鋭・heisokuが贈るリアル労働グルメ物語!

【最新版Gemini 3に対応!】できるGemini (できるシリーズ)
Geminiを「最強の知的生産パートナー」として使いこなすための、実践的なノウハウを凝縮した一冊です。 基本的な操作方法から、具体的なビジネスシーンでの活用、日々の業務を自動化するGoogle Workspaceとの連携、さらには自分だけのオリジナルAIを作成する方法まで余すところなく解説します。

Rustプログラミング完全ガイド 他言語との比較で違いが分かる!
Rustの各手法や考え方を幅広く解説! 500以上のサンプルを掲載。実行結果も確認。 全24章の包括的なチュートリアル。

ポチらせる文章術
販売サイト・ネット広告・メルマガ・ブログ・ホームページ・SNS… 全WEB媒体で効果バツグン! カリスマコピーライターが教える「見てもらう」「買ってもらう」「共感してもらう」すべてに効くネット文章術

小型で便利な Type-C アダプター USB C オス - USB3.1 オスアダプター
Type-C端子のマイコンボードをこのアダプタを介して直接Raspberry Piに挿すことができます。ケーブルなしで便利なツールです。

Divoom Ditoo Pro ワイヤレススピーカー
15W高音質重低音/青軸キーボード/Bluetooth5.3/ピクセルアート 専用アプリ/USB接続/microSDカード

電源供給USBケーブル スリム 【5本セット】
USB電源ケーブル 5V DC電源供給ケーブル スリム 【5本セット】 電源供給 バッテリー 修理 自作 DIY 電子工作 (100cm)

Copyright © 2011-2027 Sarako Tsukiyono All rights reserved®.