Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependencies: RemoteIR TextLCD
main.cpp
- Committer:
- yangtzuli
- Date:
- 2020-07-27
- Revision:
- 2:38825726cb1b
- Parent:
- 1:5bb497a38344
- Child:
- 3:2ae6218973be
File content as of revision 2:38825726cb1b:
/* mbed Microcontroller Library
* Copyright (c) 2019 ARM Limited
* SPDX-License-Identifier: Apache-2.0
*/
#include "mbed.h"
#include "ReceiverIR.h"
#include "rtos.h"
#include <stdint.h>
#include "platform/mbed_thread.h"
#include "TextLCD.h"
#define MIN_V 2.23
#define MAX_V 3.3
//bChangeのいろいろ
I2C i2c_lcd(p9, p10);
TextLCD_I2C lcd(&i2c_lcd, (0x27 << 1), TextLCD::LCD16x2, TextLCD::HD44780);
int b = 0;
AnalogIn battery(p15);
double test;
Serial pc(USBTX, USBRX);
// ポートp15を赤外線受信モジュールの接続先に指定
ReceiverIR ir_rx(p15);
RemoteIR::Format format;
uint8_t buf[32];
uint32_t bitcount;
uint32_t code;
//障害物回避の設定
DigitalOut trig(p6); // 超音波センサのtrigピンをp6に接続
DigitalIn echo(p7); // 超音波センサのechoピンをp7に接続
PwmOut servo(p25); // サーボコントロールピン(p25)
enum Mode{
ADVANCE,
RIGHT,
LEFT,
BACK,
STOP,
LINE_TRACE,
AVOIDANCE,
READY
};
Mode run;
Mode mode;
Timer timer; // 距離計測用タイマー
/* 障害物検知用の変数設定 */
int SC; // 正面
int SL; // 左
int SR; // 右
int SLD; // 左前
int SRD; // 右前
int flag = 0; // 障害物有無のフラグ
const int limit = 20; // 障害物の距離のリミット(単位:cm)
int DT; // 距離
int houkou; // 進行方向(1:前 2:左 3:右)
int far; // 最も遠い距離
int i; // ループ変数
void rimokon(void const *argument);
void avoidance(void const *argument);
void motor(void const *argument);
void watchsurrounding();
int watch();
void bChange(void const *argument);
Thread thread1(rimokon, NULL, osPriorityRealtime);
Thread thread2(motor, NULL, osPriorityHigh);
Thread thread3(avoidance, NULL, osPriorityAboveNormal);
Thread thread4(bChange, NULL, osPriorityBelowNormal);
void rimokon(void const *argument){
while(1){
// 受信待ち
if (ir_rx.getState() == ReceiverIR::Received) {
// コード受信
bitcount = ir_rx.getData(&format, buf, sizeof(buf) * 8);
if(bitcount > 1){
// 受信成功
code=0;
for(int j=0;j<4;j++){
code+=(buf[j]<<(8*(3-j)));
}
pc.printf("%0x\r\n",code);
switch(code){
case 0x40bf0ff0://入力切換
pc.printf("入力切換\r\n");
break;
case 0x40bf12ed://電源
pc.printf("電源\r\n");
break;
case 0x40bf7b84://地アナ
pc.printf("地アナ\r\n");
break;
case 0x40bf7a85://地デジ
pc.printf("地デジ\r\n");
break;
case 0x40bf7c83://BS
pc.printf("BS\r\n");
break;
case 0x40bf7d82://CS
pc.printf("CS\r\n");
break;
case 0x40bf01fe://1
pc.printf("1\r\n");
break;
case 0x40bf02fd://2
pc.printf("2\r\n");
break;
case 0x40bf03fc://3
pc.printf("3\r\n");
break;
case 0x40bf04fb://4
pc.printf("4\r\n");
break;
case 0x40bf05fa://5
pc.printf("5\r\n");
break;
case 0x40bf06f9://6
pc.printf("6\r\n");
break;
case 0x40bf07f8://7
pc.printf("7\r\n");
break;
case 0x40bf08f7://8
pc.printf("8\r\n");
break;
case 0x40bf09f6://9
pc.printf("9\r\n");
break;
case 0x40bf0af5://10
pc.printf("10\r\n");
break;
case 0x40bf0bf4://11
pc.printf("11\r\n");
break;
case 0x40bf0cf3://12
pc.printf("12\r\n");
break;
case 0x40bf1be4://チャンネル↑
pc.printf("チャンネル↑\r\n");
break;
case 0x40bf1fe0://チャンネル↓
pc.printf("チャンネル↓\r\n");
break;
case 0x40bf1ce3://画面表示
pc.printf("画面表示\r\n");
break;
case 0x40bf10ef://消音
pc.printf("消音\r\n");
break;
case 0x40bf27d8://クイック
pc.printf("クイック\r\n");
break;
case 0x40bf1ae5://音量↑
pc.printf("音量↑\r\n");
break;
case 0x40bf1ee1://音量↓
pc.printf("音量↓\r\n");
break;
case 0x40be34cb://レグザリンク
pc.printf("レグザリンク\r\n");
break;
case 0x40bf6e91://番組表
pc.printf("番組表\r\n");
break;
case 0x40bf3bc4://戻る
pc.printf("戻る\r\n");
break;
case 0x40bf3cc3://終了
pc.printf("終了\r\n");
break;
case 0x40bf3ec1://↑
pc.printf("↑\r\n");
break;
case 0x40bf3fc0://↓
pc.printf("↓\r\n");
break;
case 0x40bf5fa0://←
mode=LINE_TRACE;
pc.printf("←\r\n");
break;
case 0x40bf5ba4://→
mode=AVOIDANCE;
pc.printf("→\r\n");
break;
case 0x40bf3dc2://決定
pc.printf("決定\r\n");
break;
case 0x40bf738c://青
pc.printf("青\r\n");
break;
case 0x40bf748b://赤
pc.printf("赤\r\n");
break;
case 0x40bf758a://緑
pc.printf("緑\r\n");
break;
case 0x40bf7689://黄
pc.printf("黄\r\n");
break;
case 0x43bc14eb://dデータ
pc.printf("dデータ\r\n");
break;
case 0x40bf50af://静止
pc.printf("静止\r\n");
break;
case 0x40bf59a6://おまかせ映像
pc.printf("おまかせ映像\r\n");
break;
case 0x40bf13ec://音声切換
pc.printf("音声切換\r\n");
break;
default:
;
}
}
}
ThisThread::sleep_for(10);
}
}
void motor(void const *argument){
while(1){
pc.printf("motor\r\n");
ThisThread::sleep_for(20);
}
}
/* 障害物回避走行 */
void avoidance(void const *argument){
while(1){
if(mode==AVOIDANCE){
pc.printf("avoidance\r\n");
if(flag == 0){ // 障害物がない場合
run = ADVANCE; // 前進
}
/* else{ // 障害物がある場合
i = 0;
if(SC < 15){ // 正面15cm以内に障害物が現れた場合
run = BACK; // 後退
while(watch() < limit){ // 正面20cm以内に障害物がある間
}
run = STOP; // 停止
}
if(SC < limit && SLD < limit && SL < limit && SRD < limit && SR < limit){ // 全方向に障害物がある場合
run = LEFT; // 左折
while(i < 100){ // 進行方向確認
if(watch() > limit){
i++;
}
else{
i = 0;
}
}
run = STOP; // 停止
}
else { // 全方向以外
far = SC; // 正面を最も遠い距離に設定
houkou = 1; // 進行方向を前に設定
if(far < SLD || far < SL){ // 左または左前がより遠い場合
if(SL < SLD){ // 左前が左より遠い場合
far = SLD; // 左前を最も遠い距離に設定
}
else{ // 左が左前より遠い場合
far = SL; // 左を最も遠い距離に設定
}
houkou = 2; // 進行方向を左に設定
}
if(far < SRD || far < SR){ // 右または右前がより遠い場合
if(SR < SRD){ // 右前が右より遠い場合
far = SRD; // 右前を最も遠い距離に設定
}
else{ // 右が右前よりも遠い場合
far = SR; // 右を最も遠い距離に設定
}
houkou = 3; // 進行方向を右に設定
}
switch(houkou){ // 進行方向の場合分け
case 1: // 前の場合
run = ADVANCE; // 前進
thread_sleep_for(100); // 1秒待つ
break;
case 2: // 左の場合
run = LEFT; // 左折
while(i < 100){ // 進行方向確認
if(watch() > (far - 2)){
i++;
}
else{
i = 0;
}
}
run = STOP; // 停止
break;
case 3: // 右の場合
run = RIGHT; // 右折
while(i < 100){ // 進行方向確認
if(watch() > (far - 2)){
i++;
}
else{
i = 0;
}
}
run = STOP; // 停止
break;
}
}
}*/
flag = 0; // フラグを0にセット
watchsurrounding();
}
}
}
int watch(){
pc.printf("watch\r\n");
trig = 0;
ThisThread::sleep_for(5); // 5ms待つ
trig = 1;
ThisThread::sleep_for(15); // 15ms待つ
trig = 0;
//while(echo.read() == 0){
//}
timer.start(); // 距離計測タイマースタート
while(echo.read() == 1){
}
timer.stop(); // 距離計測タイマーストップ
DT = timer.read_us()*0.01657; // 距離計算
if(DT > 400){ // 検知範囲外なら400cmに設定
DT = 400;
}
timer.reset(); // 距離計測タイマーリセット
ThisThread::sleep_for(30); // 30ms待つ
return DT;
}
void watchsurrounding(){
pc.printf("watchsurrounding\r\n");
SC = watch();
if(SC < limit){ // 正面20cm以内に障害物がある場合
run = STOP; // 停止
}
servo.pulsewidth_us(1925); // サーボを左に40度回転
ThisThread::sleep_for(250); // 250ms待つ
SLD = watch();
if(SLD < limit){ // 左前20cm以内に障害物がある場合
run = STOP; // 停止
}
servo.pulsewidth_us(2400); // サーボを左に90度回転
ThisThread::sleep_for(250); // 250ms待つ
SL = watch();
if(SL < limit){ // 左20cm以内に障害物がある場合
run = STOP; // 停止
}
servo.pulsewidth_us(1450);
ThisThread::sleep_for(100);
servo.pulsewidth_us(925); // サーボを右に40度回転
ThisThread::sleep_for(250); // 250ms待つ
SRD = watch();
if(SRD < limit){ // 右前20cm以内に障害物がある場合
run = STOP; // 停止
}
servo.pulsewidth_us(500); // サーボを右に90度回転
ThisThread::sleep_for(250); // 250ms待つ
SR = watch();
if(SR < limit){ // 右20cm以内に障害物がある場合
run = STOP; // 停止
}
servo.pulsewidth_us(1450); // サーボを中央位置に戻す
ThisThread::sleep_for(100); // 100ms待つ
if(SC < limit || SLD < limit || SL < limit || SRD < limit || SR < limit){ // 20cm以内に障害物を検知した場合
flag = 1; // フラグに1をセット
}
}
void bChange(void const *argument){
while(1){
//lcd.setBacklight(TextLCD::LightOn);
test = battery.read()*MAX_V;
b = (int)((test - MIN_V)/0.107 + 0.5)*10;
lcd.locate(0,0);
lcd.printf("Battery:%3d%%",b);
pc.printf("bChange\r\n");
}
}
int main() {
while(1){
/*if(button.read()==0){
rimokon();
} */
pc.printf("main\r\n");
ThisThread::sleep_for(40);
}
}