ラベル センサー の投稿を表示しています。 すべての投稿を表示
ラベル センサー の投稿を表示しています。 すべての投稿を表示

2022年8月16日火曜日

SCCBバスプロトコル詳解と実装方法

 ■SCCB紹介

 SCCB (Serial Camera Control Bus) は、OmniVision社のシリアル カメラ バス プロトコルです。OV7670 などの OV で始まるカメラ モジュールは、SCCB プロトコルを使用します。I2C プロトコルと類似しているため、SIO_CSIO_DSCLSDAと呼びます。

 本文は2線式SCCBプロトコルの内容とプログラム実装を中心に解説します。

■回路図

 2線式SCCBに対して、SIO_CSIO_Dだけを接続すればよいです。

■プロトコルの内容とプログラム実装

2 線式 SCCB バス プロトコル解析

<1>送信開始フラグ: 送信 (データの読み取りまたは書き込み) ごとに、下図に従って SIO_C SIO_D に指定されたHIGH-LOWレベルを出力する必要があります。対応するレベルを受け取ると、Slaveは転送開始が判ります。次のように操作します。SIO_CHIGHのままでSIO_DHIGHからLOWに変換して、SIO_CLOWに戻ります。

void SCCB_Start(void) {

  SCCB_SDA = 1;

  SCCB_SCL = 1;

  delay_us(50);

  SCCB_SDA = 0;

  delay_us(50);

  SCCB_SCL = 0;

}

<2>送信完了フラグ:Slaveに送信完了を通知します。次のように操作します。SIO_CHIGHのままでSIO_DLOWからHIGHに変換します。 

void SCCB_Stop(void) {

  SCCB_SDA = 0;

  delay_us(50);

  SCCB_SCL = 1;

  delay_us(50);

  SCCB_SDA = 1;

  delay_us(50);

}

<3>伝送フェーズ

転送フェーズ(Transmission Phases)は、データ転送の基本単位です。9bitで構成しています、それぞれは、bit7bit09bit目のDon't careまたはNAです。Don't care ビットと NA ビットの意味は後で説明します。

<4>書き込み

Slave1バイトのデータを書き込みすることは1回書き込み操作と呼びます。1バイトのデータを書き込み時、SlaveID、対象レジスタ番号も伝える必要があるため、書き込み操作のプロセスは三つのフェーズ(Phase)が必要です。

フェーズ1SCCBプロトコルは複数のSlaveに転送ができるため、Slaveの番号が伝える必要があります。実際のIDアドレスは8bitで、bit7bit1Slaveの番号ですが、0127のトータル128Slaveに転送が可能です。bit0Slaveにデータを書き込み(0)か読み取り(1)かを示しています。9bit目はDon't careビット(上図Xのビット)です。例えば、OV7670の場合、Masterから 8bitのデータを受信した後、SIO_C=1の期間にSlaveSIO_DピンにLOWをセットします。この間、MasterSIO_Dのレベル情報を読み取って判断する、LOWレベルを読み取った場合は、Slaveはフェーズ1の最初8bitを正常受信したことを示して、送信は成功します。それ以外の場合、送信は失敗します。

フェーズ2:データが書き込まれるレジスタの番号をスレーブに送信します。レジスタの番号は、OVセンサーのデータシートに記載されています。レジスタの番号も 8bitのデータです。 同様に、フェーズ29bit目もDon't careビットであり、このbitの説明はフェーズ1と同じです。

フェーズ3:指定されたレジスタにデータを書き込みます。bit7bit0は、レジスタに書き込みたいデータです。9ビット目もDon't careビットです。

u8 SCCB_WR_Byte(u8 dat) { // 一つフェーズを書き込み

  u8 j, res;

  for (j = 0; j < 8; j++) {

    if (dat&0x80) SCCB_SDA = 1;

    else SCCB_SDA = 0;

    data<<=1;

    delay_us(50);

    SCCB_SCL = 1;

    delay_us(50);

    SCCB_SCL = 0;

}

SCCB_SDA_IN();  // 或いはpinMode(SDA, INPUT); Slaveは値をセット可能

delay_us(50);

SCCB_SCL = 1; // SCLHIGHにする、Slaveはデータを受信した後、SDALOW

delay_us(50);

if (SCCB_READ_SDA) res = 1;  // SDAHIGHの場合、送信失敗

  else res = 0;  // 送信成功

  SCCB_SCL = 0;

  SCCB_SDA_OUT(); // あるいは pinMode(SDA, OUTPUT);

  Return res;

}

三つフェーズを書き込みの関数:

u8 SCCB_WR_Reg(u8 reg, u8 data) {

  u8 res = 0;

  SCCB_Start();

  If (SCCB_WR_Byte(SCCB_ID)) res = 1; // OV7670の場合は0x42

  delay_us(100);

  if (SCCB_WR_Byte(reg)) res = 1;

  delay_us(100);

  if (SCCB_WR_Byte(data)) res = 1;

  SCCB_Stop();

  return res;

} 

<5>読み取り操作

Slaveのレジスタから 1 バイトのデータを読み取る時、次の 2 つのデータ送信プロセスが必要です。

2 フェーズ書き込み送信プロセス:MasterからSlave2フェーズを送信します。最初のフェーズは、スレーブの ID 番号を示します。 2 番目のフェーズは、スレーブのどのレジスタから読み取るかを示します。 

2フェーズ読み取り送信プロセス:Slaveからデータを読み取っています。フェーズ1は①のフェーズ1とほぼ同じですが、Slaveから読み取りから、bit01のほうをご注意ください。フェーズ2は転送データです。

したがって、SIO_Dにつながっているピンは入力モードに設定する必要があります。8bitのデータを読み取った後、データが正常に受信されたことをSlaveに伝えるために、Master1bitの応答信号(NA)Slaveに送信する必要があります。

NAを送信するために、MasterSIO_DHIGHにセットしたままで、SIO_CLOWからHIGHに変更してからLOWに戻る必要があります。

void SCCB_No_Ack(void) {

  delay_us(50);

  SCCB_SDA = 1;

  SCCB_SCL = 1;

  delay_us(50);

  SCCB_SCL = 0;

  delay_us(50);

  SCCB_SDA = 0;

  delay_us(50);

}

2フェーズを読み取り関数

u8 SCCB_RD_Byte(void) {

  u8 temp = 0, j;

  SCCB_SDA_IN();

  for (j = 8; j > 0; j--) {

    delay_us(50);

    SCCB_SCL = 1;

    temp = temp << 1;

    if (SCCB_READ_SDA) temp++;

    delay_us(50);

    SCCB_SCL = 0;

  }

  SCCB_SDA_OUT();

  return temp;

}

レジスタから読み取り関数:

u8 SCCB_RD_Reg(u8 reg) {

  u8 val = 0;

  SCCB_Start();

  SCCB_WR_Byte(SCCB_ID);

  delay_us(100);

  SCCB_WR_Byte(reg);

  delay_us(100);

  SCCB_Stop();

  

  SCCB_Start();

  SCCB_WR_Byte(SCCB_ID|0x01);

  delay_us(100);

  val = SCCB_RD_Byte();

  SCCB_No_Ack();

  SCCB_Stop();

  return val;

}

ここまで、SCCBのレジスタ読み取り操作とレジスタ書き込み操作の関数SCCB_RD_RegSCCB_WR_Regを実装しました。この二つ関数を利用して、OVセンサーの指定されたレジスタへのデータを読み取りしたり、書き込みしたり、センサーの初期化ができました。



2022年7月31日日曜日

WebSocketでOV7670-ESP32の画像表示

 ■OV7670

 OV7670 はVGA(640×480画素)サイズのイメージセンサです。内部構造はCMOSイメージセンサと信号処理用 DSPがワンチップになっています。マイコンへの映 像信号出力は、ディジタル8ビット・パラレルです。
製品仕様:
・イメージセンサー:OV7670 CMOS、VGA-640(V)×480(V)
・レンズ焦点距離:3.6mm(F2.5)
・光学サイズ:1/6型、画素セル:3.6um x 3.6um
・フィールド角:25°(対角)
・小さいサイズ:3.5cm x 3.5cm
・低動作電圧:2.5V~3.0VDC
・低動作電流:60mw(15fps VGA YUV format)
・スリープモード<20μA
・保存温度:-30℃~70℃
・動作温度:0℃~50℃
・感度:1.3V/(Lux-sec)
・インターフェース:標準SCCB(Serial Camera Control Bus)インターフェース(I2C互換)
・データ出力フォーマット:Raw RGB、RGB(GRB4:2:2)、RGB(565/555/444)、YUV(4:2:2)、YCbCr(4:2:2)
・データ出力解像度:VGA、CIF、CIF~40x30の任意解像度
・サンプリングタイプ:Vario Pixel
・接続コネクタ:2.54mmピッチ(2列)18ピン、ピンヘッダ
■ESP32接続
 下記ように、OV7670のピンとESP32ピンを接続します。
 ※データ転送スピードを考えると、短いジャンプワイヤーが勧め
 ※SDAとSCLは1KΩの抵抗を使用でも可
OV7670ESP32
D0IO36
D1IO39
D2IO34
D3IO35
D4IO32
D5IO33
D6IO25
D7IO26
MCLKIO15
PCLKIO14
HREFIO23
VSYNCIO13
SDAIO21 (1kΩ Pull Up)
SCLIO22 (1KΩ Pull Up)
RST3.3V
PWDNGND
■Web送信
 Web送信はWebSocketを使いました。
 下記ライブラリが必要です。
■部品一覧:
・ESP-WROOM-32開発ボード
・OV7670
・USBケーブル
・ジャンプワイヤー
・1KΩ抵抗*2
・ブレッドボード
■プログラム:
 単純なESP32、OV7670と接続ため(FIFOなどがない)、インターネットのいろいろなサンプルを試した結果で、
は使えます。
 
 下記箇所をちょっと修正しました。
 ①13行目~14行目までのWiFi接続情報
 ②23行目~33行目までのピン情報
 ③APモードは不要のため、削除
 ④DBABuffer.hはコンパイルエラーあるため、「#include <stdlib.h>」を追加

#include "OV7670.h"

#include <WebSockets.h>
#include <WebSocketsClient.h>
#include <WebSocketsServer.h>
#include <WiFi.h>
#include <WiFiMulti.h>
#include <WiFiClient.h>
#include "canvas_htm.h"

const char *ssid_AP_1 = "XXXXXX";
const char *pwd_AP_1  = "XXXXXXX";

const int SIOD = 21; //SDA
const int SIOC = 22; //SCL

const int VSYNC = 13;
const int HREF = 23;

const int XCLK = 15;
const int PCLK = 14;

const int D0 = 36;
const int D1 = 39;
const int D2 = 34;
const int D3 = 35;
const int D4 = 32;
const int D5 = 33;
const int D6 = 25;
const int D7 = 26;

const int TFT_DC = 2;
const int TFT_CS = 5;

OV7670 *camera;

WiFiMulti wifiMulti;
WiFiServer server(80);

unsigned char pix = 0;

unsigned char start_flag = 0xAA;
unsigned char end_flag = 0xFF;
unsigned char ip_flag = 0x11;

WebSocketsServer webSocket(81);    // create a websocket server on port 81

void startWebSocket() { // Start a WebSocket server
  webSocket.begin();                          // start the websocket server
  webSocket.onEvent(webSocketEvent);          // if there's an incomming websocket message, go to function 'webSocketEvent'
  Serial.println("WebSocket server started.");
}

void startWebServer()
{
   server.begin();
   Serial.println("Http web server started.");
}
void serve() {
  WiFiClient client = server.available();
  if (client) 
  {
    //Serial.println("New Client.");
    String currentLine = "";
    while (client.connected()) 
    {
      if (client.available()) 
      {
        char c = client.read();
        //Serial.write(c);
        if (c == '\n') 
        {
          if (currentLine.length() == 0) 
          {
            client.println("HTTP/1.1 200 OK");
            client.println("Content-type:text/html");
            client.println();
            client.print(canvas_htm);
            client.println();
            break;
          } 
          else 
          {
            currentLine = "";
          }
        } 
        else if (c != '\r') 
        {
          currentLine += c;
        }
        
      }
    }
    // close the connection:
    client.stop();

  }  
}

void webSocketEvent(uint8_t num, WStype_t type, uint8_t * payload, size_t payloadlength) { // When a WebSocket message is received
 
  int blk_count = 0;

  char canvas_VGA[] = "canvas-VGA";
  char canvas_Q_VGA[] = "canvas-Q-VGA";
  char canvas_QQ_VGA[] = "canvas-QQ-VGA";
  char canvas_QQQ_VGA[] = "canvas-QQQ-VGA";
  char ipaddr[26] ;
  IPAddress localip;
  
  switch (type) {
    case WStype_DISCONNECTED:             // if the websocket is disconnected
      Serial.printf("[%u] Disconnected!\n", num);
      break;
    case WStype_CONNECTED: {              // if a new websocket connection is established
        IPAddress ip = webSocket.remoteIP(num);
        Serial.printf("[%u] Connected from %d.%d.%d.%d url: %s\n", num, ip[0], ip[1], ip[2], ip[3], payload);
           webSocket.sendBIN(0, &ip_flag, 1);
           localip = WiFi.localIP();
           sprintf(ipaddr, "%d.%d.%d.%d", localip[0], localip[1], localip[2], localip[3]);
           webSocket.sendTXT(0, (const char *)ipaddr);
           
      }
      break;
    case WStype_TEXT:                     // if new text data is received
      if (payloadlength == sizeof(canvas_QQQ_VGA)-1) {
        if (memcmp(canvas_QQQ_VGA, payload, payloadlength) == 0) {
              Serial.printf("canvas_QQQ_VGA");
              webSocket.sendBIN(0, &end_flag, 1);
              camera = new OV7670(OV7670::Mode::QQQVGA_RGB565, SIOD, SIOC, VSYNC, HREF, XCLK, PCLK, D0, D1, D2, D3, D4, D5, D6, D7);
        }
      } else if (payloadlength == sizeof(canvas_QQ_VGA)-1) {
        if (memcmp(canvas_QQ_VGA, payload, payloadlength) == 0) {
              Serial.printf("canvas_QQ_VGA");
              webSocket.sendBIN(0, &end_flag, 1);
              camera = new OV7670(OV7670::Mode::QQVGA_RGB565, SIOD, SIOC, VSYNC, HREF, XCLK, PCLK, D0, D1, D2, D3, D4, D5, D6, D7);
        }
      } else if (payloadlength == sizeof(canvas_Q_VGA)-1) {
        if (memcmp(canvas_Q_VGA, payload, payloadlength) == 0) {
              Serial.printf("canvas_Q_VGA");
              webSocket.sendBIN(0, &end_flag, 1);
              camera = new OV7670(OV7670::Mode::QVGA_RGB565, SIOD, SIOC, VSYNC, HREF, XCLK, PCLK, D0, D1, D2, D3, D4, D5, D6, D7);
        }
      } else if (payloadlength == sizeof(canvas_VGA)-1) {
        if (memcmp(canvas_VGA, payload, payloadlength) == 0) {
              Serial.printf("canvas_VGA");
              webSocket.sendBIN(0, &end_flag, 1);
              camera = new OV7670(OV7670::Mode::VGA_RGB565, SIOD, SIOC, VSYNC, HREF, XCLK, PCLK, D0, D1, D2, D3, D4, D5, D6, D7);
        }
      } 

      
      blk_count = camera->yres/I2SCamera::blockSlice;//30, 60, 120
      for (int i=0; i<blk_count; i++) {

          if (i == 0) {
              camera->startBlock = 1;
              camera->endBlock = I2SCamera::blockSlice;
              webSocket.sendBIN(0, &start_flag, 1);
          }

          if (i == blk_count-1) {
              webSocket.sendBIN(0, &end_flag, 1);
          }
        
          camera->oneFrame();
          webSocket.sendBIN(0, camera->frame, camera->xres * I2SCamera::blockSlice * 2);
          camera->startBlock += I2SCamera::blockSlice;
          camera->endBlock   += I2SCamera::blockSlice;
      }
      
      break;
    case WStype_ERROR:                     // if new text data is received
      Serial.printf("Error \n");
    default:
      Serial.printf("WStype %x not handled \n", type);

  }
}

void initWifiMulti()
{

    wifiMulti.addAP(ssid_AP_1, pwd_AP_1);

    Serial.println("Connecting Wifi...");
    while(wifiMulti.run() != WL_CONNECTED) {
       delay(5000);        
       Serial.print(".");
    }
    
    Serial.print("\n");
    Serial.print("WiFi connected : ");
    Serial.print("IP address : ");
    Serial.println(WiFi.localIP());
}

void setup() {
  Serial.begin(115200);
  initWifiMulti();
  startWebSocket();
  startWebServer();
}

void loop()
{
  webSocket.loop();
  serve();
}
■結果
ブラウザーを開いて、URLを入力し
配線イメージ
写真工場 :)

2022年7月28日木曜日

リードスイッチ形磁気形近接センサー、静電容量方式タッチセンサー、アクティブブザーモジュールの連動

■リードスイッチ形磁気形近接センサー


 リードスイッチは下図のように2枚のニッケル合金(磁性体)リードの一部をオーバーラップさせ、ガラス管に封入したもので、 このオーバーラップ部分は表面に金、銀及びロジウムなどの接点処理が施されており、さらに接点の活性化を防ぐためにガラス管内に不活性ガスを封入している。

 リードスイッチに電磁コイルまたは磁石によって磁界が作用するとリードが磁化され、磁界の2乗に比例したオーバーラップ部分の吸引力Fpが生じる。

 この吸引力でリードがたわむと、リード自体の弾性により反発力Fsが生じる。

 すなわち、Fp>Fsであるような磁界において接点は閉成される。

 また、Fp<Fsとなるような磁界において接点は開放される。

 磁気形近接スイッチでは、このようなリードスイッチの動作機構を利用し、磁石との組合せにおいて各種の機能を得ている。

■静電容量方式タッチセンサー

 「タッチセンサー」の表面に指で触れると生じる静電容量(電荷)の変化を、「センサー」が感知することで位置を認識します。

 「静電容量」とは、電気を蓄える能力であり、人体など、電気を通すものが近づいただけで変化します。

 電気を蓄えられる時間の変化から「静電容量」の増減を調べられるので、人体の接近を検知することが出来ます。

 この「センサー」は、指と「センサー」の間に遮蔽物があっても機能します。

■アクティブブザーモジュール

 電気を流すだけでピーと音が鳴ります。digitalWrite() で ON/OFF して制御します。

■磁気検知機

 磁気検知機が作成られると思って、下記機能を想定されています。

 磁石を近くにより、磁気を感知され、アクティブブザーを発音します。タッチセンサーにタッチして、消音します。

 見やすいために、赤LEDを追加し、アクティブブザーを発音/消音に従って、点灯/消灯します。

■配線図

※実物のアクティブブザーモジュールの「+」「S」ピンは間違いました。

 正しい順番は「-」「+」「S」

■プログラム:

#define BUZZER 4  // D2
#define TOUCH_PIN 5  // D1
#define REED 2  // D4
#define LED_PIN 16  // D0
void setup() {
  Serial.begin(115200);
  pinMode(BUZZER, OUTPUT);
  pinMode(LED_PIN, OUTPUT);
  pinMode(TOUCH_PIN, INPUT);
  pinMode(REED, INPUT);
  digitalWrite(BUZZER, LOW);
  digitalWrite(LED_PIN, LOW);
}
void loop() {
  int val = digitalRead(REED);
  Serial.print("val=");
  Serial.println(val);
  if (val == LOW) {
    // 磁気近くになる時
    Serial.println("Scanned!");
    digitalWrite(BUZZER, HIGH);
    digitalWrite(LED_PIN, HIGH);
  } else {
    // 磁気が遠くになる時
    digitalWrite(BUZZER, LOW);
    digitalWrite(LED_PIN, LOW);
  }
  if (digitalRead(TOUCH_PIN) == HIGH) {
    // タッチの時
    Serial.println("Cleared!");
    digitalWrite(BUZZER, LOW); 
    digitalWrite(LED_PIN, LOW);
  }
  delay(1000);
}

■実物イメージ

磁気近くなる時:


 タッチセンサーにタッチする:

予想通り :)

2022年7月25日月曜日

温度と湿度センサー(DHT11)の使い方

 ■DHT11の紹介

  DHT11は、温度と湿度を同時に計測することができるセンサーです。下記はモジュール版となり、DHT11単体で使おうとするとデータ受信ピンのプルアップが必要になるのですが、モジュールの場合はそのような対応は不要で、そのままArduinoに接続すれば使えるようになるので便利です。


    DHT11の仕様:

 入力電圧:DC 3~5.5V
 信号出力タイプ:デジタル
 温度測定範囲:0 ℃ ~ 50 ℃
 湿度測定分解能:1℃
 温度測定誤差範囲:±2℃
 湿度測定範囲:20%~90%RH
 湿度測定分解能:1%RH
 湿度測定誤差範囲:±5%RH

■配線図
■部品一覧
ESP8266開発ボード
・DHT11センサーモジュール
USBケーブル
ブレッドボード
ジャンパーワイヤー

■プログラム:
事前に、ライブラリ「DHT_sensor_library」をインストールしてください。

#include "DHT.h"
#define DHTTYPE DHT11   // DHT 11

#define dht_dpin 5
DHT dht(dht_dpin, DHTTYPE);

void setup() {
  Serial.begin(115200);
  dht.begin();
  Serial.println("Humidity and temperature\n\n");
}

void loop() {
  float h = dht.readHumidity();
  float t = dht.readTemperature();         
  Serial.print("Current humidity = ");
  Serial.print(h);
  Serial.print("%  ");
  Serial.print("temperature = ");
  Serial.print(t); 
  Serial.println("C  ");
  delay(800);
}

■ 実物イメージ:

■シリアルモニタ出力



2022年7月16日土曜日

声感センサー

  今日の話題は一つ安い声感センサー。該当センサーは声の識別ができない、声が有無しか感知できます。声感知スイッチなど場合は適用です。

  ものはこれです。

■配線図


■部品リスト
ESP8266開発ボード
・声感センサー
ブレッドボード
USBケーブル
ジャンパーワイヤー
・抵抗

■プログラム
#define SOUND A0
void setup() {
  Serial.begin(115200);
  pinMode(SOUND, INPUT);
}
void loop() {
  Serial.println(analogRead(SOUND));
  
  delay(1000);
}

■結果

    感度は青半固定ボリュームで動的調整できます。

 シリアルモニタの出力結果


 実物イメージ:


  実験結果によって、やっぱり、該当センサーは声の識別がだめです。


シンプルロボットカーセットの使い方

■ マイコン :ESP3266-CH340  シンプルの案として、マイコンは安いESP8266-CH340開発ボードを選びました。該当マイコンもWIFI機能があります。  ドライバーダウンロード場所:    https://sparks.gogo.co.nz/ch340.html...