明石高専ロボ研 drum

Dependencies:   mbed

scrp_slave.cpp

Committer:
TanakaRobo
Date:
2020-01-02
Revision:
0:ca84ed7518f5
Child:
2:141358d84ff4

File content as of revision 0:ca84ed7518f5:

#include "scrp_slave.hpp"

#define STX 0x41
#define DMY 0xff

ScrpSlave::ScrpSlave(PinName TX1,PinName RX1,uint32_t addr):address_(addr){
    mode_ = 0;
    init(TX1,RX1);
}

ScrpSlave::ScrpSlave(PinName TX1,PinName RX1,PinName REDE1,uint32_t addr):address_(addr){
    mode_ = 1;
    rede_ = new DigitalOut(REDE1);
    init(TX1,RX1);
}

ScrpSlave::ScrpSlave(PinName TX1,PinName RX1,PinName TX2,PinName RX2,uint32_t addr):address_(addr){
    mode_ = 2;
    serial_[1] = new BufferedSerial(TX2,RX2);
    serial_[1]->baud(115200);
    serial_[1]->attach(callback(this,&ScrpSlave::port2),Serial::RxIrq);
    init(TX1,RX1);
}

ScrpSlave::ScrpSlave(PinName TX1,PinName RX1,PinName REDE1,PinName TX2,PinName RX2,uint32_t addr):address_(addr){
    mode_ = 3;
    rede_ = new DigitalOut(REDE1);
    serial_[1] = new BufferedSerial(TX2,RX2);
    serial_[1]->baud(115200);
    serial_[1]->attach(callback(this,&ScrpSlave::port2),Serial::RxIrq);
    init(TX1,RX1);
}

void ScrpSlave::init(PinName TX,PinName RX){
    timeout_ = 10;
    serial_[0] = new BufferedSerial(TX,RX);
    serial_[0]->baud(115200);
    serial_[0]->attach(callback(this,&ScrpSlave::port1),Serial::RxIrq);
    flash_ = new FlashIAP;
    if(flash_->init()==0){
        if(flash_->read(&my_id_,address_,1) != 0){
            send(222,222,222);
            my_id_ = 10;
        }
    }else{
        send(111,111,111);
        my_id_ = 10;
    }
    for(int i = 1;i<255;++i){
        procs_[i] = NULL;
    }
}

void ScrpSlave::port1(){
    check(0);
}

void ScrpSlave::port2(){
    check(1);
}

void ScrpSlave::addCMD(uint8_t cmd, bool (*proc)(int rx_data, int& tx_data)){
    if(cmd == 0 || cmd == 254 || cmd == 253)return;
    procs_[cmd] = proc;
}

void ScrpSlave::setTimeout(int time){
    timeout_ = time;
}

void ScrpSlave::changeID(uint8_t id){
    flash_->erase(address_,flash_->get_sector_size(address_));
    flash_->program(&id,address_,1);
}

int16_t ScrpSlave::send(uint8_t id,uint8_t cmd,int16_t tx_data){
    return sending(0,id,cmd,tx_data);
}

int16_t ScrpSlave::send2(uint8_t id,uint8_t cmd,int16_t tx_data){
    if(mode_ < 2)return -1;
    return sending(1,id,cmd,tx_data);
}

int16_t ScrpSlave::sending(int port,uint8_t id,uint8_t cmd,int16_t tx_data){
    uint8_t tx_dataL = tx_data;
    uint8_t tx_dataH = tx_data >> 8;
    uint8_t tx_sum = id + cmd + tx_dataL + tx_dataH;

    const uint8_t data[] = {DMY, STX, id, cmd, tx_dataL, tx_dataH, tx_sum, DMY};
    if(!serial_[port]->writeable()){
        return -1;
    }
    if(mode_%2 == 1 && port == 0){
        rede_->write(1);
    }
    serial_[port]->write(data,8);
    wait_ms(1);//送信待ち
    if(mode_%2 == 1 && port == 0){
        rede_->write(0);
    }
            
    int i = 0;
    bool received = false;
    bool stxflag = false;
    uint8_t getdata;
    uint8_t rx[5]={},sum = 0;
    Timer out;
    out.start();
    while(out.read_ms() < timeout_ && !received){
        while(serial_[port]->readable() > 0){
            getdata = serial_[port]->getc();
            if(!stxflag && getdata == STX){
                stxflag = true;
                continue;
            }
            if(stxflag){
                rx[i] = getdata;
                sum += rx[i++];
            }
            if(i > 4){/*
                uint8_t sum = 0;
                for(int j = 0;j<4;j++){
                    sum += rx[j];
                }*/
                if(sum == rx[4]){
                    received = true;
                }
                break;
            }
        }
    }
    out.stop();
    if(!received){
        return -1;
    }
    return (int16_t)(rx[2] + ((int16_t)rx[3] << 8));
}

void ScrpSlave::check(int port){
    uint8_t rx_cmd;
    int16_t rx_data;
    bool received = false;
    bool broadcast = false;
    while(serial_[port]->readable() >= 6){
        if(serial_[port]->getc() != STX)continue;
        uint8_t rx_id = serial_[port]->getc();
        uint8_t tmp_rx_cmd = serial_[port]->getc();
        uint8_t tmp_rx_dataL = serial_[port]->getc();
        uint8_t tmp_rx_dataH = serial_[port]->getc();
        uint8_t rx_sum = serial_[port]->getc();
        
        uint8_t sum = rx_id + tmp_rx_cmd + tmp_rx_dataL + tmp_rx_dataH;
        if(sum != rx_sum){
            continue;
        }
        
        if(rx_id == 255){
            broadcast = true;
        }else if(my_id_ == rx_id){
            broadcast = false;
        }else{
            return;//break;
        }
        
        rx_cmd = tmp_rx_cmd;
        rx_data = tmp_rx_dataL + ((int16_t)tmp_rx_dataH << 8);
        received = true;
    }
    if(!received){
        return;
    }
    int tx_data = rx_data;
    if(rx_cmd == 0){//通信テスト
        tx_data = rx_data;
    }else if(rx_cmd == 254){//id変更
        uint8_t new_id = rx_data;
        my_id_ = new_id;
        changeID(new_id);
    }else if(rx_cmd == 253){//id確認
        tx_data = my_id_;
        rx_cmd = 250;
        broadcast = false;
    }else if(procs_[rx_cmd] == NULL || !procs_[rx_cmd](rx_data,tx_data)){
        return;
    }
    if(broadcast){
        return;
    }
    
    uint8_t tx_dataL = tx_data;
    uint8_t tx_dataH = tx_data >> 8;
    uint8_t tx_sum = my_id_ + rx_cmd + tx_dataL + tx_dataH;
    
    const uint8_t data[] = {DMY, STX, my_id_, rx_cmd, tx_dataL, tx_dataH, tx_sum, DMY};
    if(!serial_[port]->writeable()){
        return;
    }
    if(mode_%2 == 1 && port == 0){
        rede_->write(1);
    }
    serial_[port]->write(data,8);
    wait_ms(1);//1ms待つ多分これだけで送信しきれる。
    if(mode_%2 == 1 && port == 0){
        rede_->write(0);
    }
    return;
}

ScrpSlave::~ScrpSlave(){
    delete serial_[0];
    delete flash_;
    if(mode_%2 == 1){
        delete rede_;
    }
    if(mode_ >= 2){
        delete serial_[1];
    }
}