hl1200aa 发表于 2024-3-5 19:01:34

慢搓PELCO_D解码板

最近想做个远程浇水的摄像头,以便管理老家的瓜瓜果果,斥巨资160买了个安佳MTJ40D摄像头模组,让店家刷了可以输出PELCO_D云台协议的固件,
可是PELCO_D云台协议的解码板实在太贵,都在200+,决定自己慢搓。
今天模组到手,组装起来。



解码还是用STC8H8K64U, 24MHZ晶振,
串口1用于ISP和仿真
串口2
RXD2 用于接收摄像头发过来的PELCO_D协议数据,
TXD2 用于向STC-ISP发送收到的数据
波特率设置为9600_8_N_1
目前测试正常



下一步慢慢研究PELCO_D协议,对接收的数据进行解码,输出相应动作。

21cnsound 发表于 2024-3-5 20:25:34

有实践、应用,才能更好掌握。
话说,远程浇水用摄像头干什么呢?能否大概介绍一下原理

hl1200aa 发表于 2024-3-5 21:49:19

21cnsound 发表于 2024-3-5 20:25
有实践、应用,才能更好掌握。
话说,远程浇水用摄像头干什么呢?能否大概介绍一下原理 ...

可以巡查地里的情况,看作物生长得怎么样了,手动浇水。

例如这个摄像头的上下左右键对应4个喷头,看到农作物有点旱的话直接按云台上的方向键喷水就可以了。




如果用在养鸡养鸽子,那就是把玉米,稻谷,豆粕等饲料加到食槽。

21cnsound 发表于 2024-3-5 21:56:11

hl1200aa 发表于 2024-3-5 21:49
可以巡查地里的情况,看作物生长得怎么样了,手动浇水。

例如这个摄像头的上下左右键对应4个喷头,看到 ...

通过单片机远程控制云台、水管电磁阀门,通过摄像头观察是否需要浇水,是吧?
如果是这样,还可以进一步,在土壤中安装土壤湿度传感器自动控制电磁阀门。

hl1200aa 发表于 2024-3-6 08:22:28

21cnsound 发表于 2024-3-5 21:56
通过单片机远程控制云台、水管电磁阀门,通过摄像头观察是否需要浇水,是吧?
如果是这样,还可以进一步 ...

那是进一步的事情了,先搞好远程手控先。

hl1200aa 发表于 2024-3-6 16:35:00

PELCO_D协议


hl1200aa 发表于 2024-3-6 16:53:51


The format for a message is:

Byte 1             Byte 2         Byte 3                Byte 4                Byte 5      Byte 6      Byte 7
Synch Byte   Address      Command 1       Command 2       Data 1      Data 2      Check Sum



All values below are shown in hexadecimal (base 16).
The synchronization byte is always $FF.


The address is the logical address of the receiver/driver being controlled.
The check sum is the 8 bit (modulo 256) sum of the payload bytes (bytes 2 through 6) in the message.


消息数据格式
字节 1                字节 2      字节 3         字节 4       字节 5   字节 6   字节 7
同步码(0xff)   地址         指令1             指令2      数据1      数据2      校验和

所有的数据以16进制显示
同步码为0xff
地址是被控制接收器/驱动器的逻辑地址
校验和是消息负载位(2-6字节)8位(模256)和。







hl1200aa 发表于 2024-3-6 18:54:57

使用实验箱的例程改出接收程序,正常接收到协议数据,并且能保存到数组。


/*---------------------------------------------------------------------*/
/* --- STC MCU Limited ------------------------------------------------*/
/* --- STC 1T Series MCU Demo Programme -------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966 ------------------------*/
/* --- Web: www.STCAI.com ---------------------------------------------*/
/* --- Web: www.STCMCUDATA.com---------------------------------------*/
/* --- BBS: www.STCAIMCU.com-----------------------------------------*/
/* --- QQ:800003751 -------------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序      */
/*---------------------------------------------------------------------*/


/*************功能说明    **************

本例程基于STC8H8K64U为主控芯片的实验箱9进行编写测试,STC8G、STC8H系列芯片可通用参考.

串口2全双工中断方式收发通讯程序。

通过PC向MCU发送数据, MCU收到后通过串口2把收到的数据原样返回.

用定时器做波特率发生器,建议使用1T模式(除非低波特率用12T),并选择可被波特率整除的时钟频率,以提高精度。

下载时, 选择时钟 22.1184MHz (用户可自行修改频率).

******************************************/

#include "../comm/stc8h.h"
#include "intrins.h"
#include "pelco_d.h"

#define MAIN_Fosc       24000000L   //定义主时钟(精确计算115200波特率)

typedef   unsigned char   u8;
typedef   unsigned int    u16;
typedef   unsigned long   u32;


#define Baudrate2   9600L
#define UART2_BUF_LENGTH    64


u8TX2_Cnt;    //发送计数
u8RX2_Cnt;    //接收计数
bit B_TX2_Busy; //发送忙标志

u8xdata RX2_Buffer; //接收缓冲

u8 pelco_d_num = 0;
u8 pelco_d_r = 0;
u8 pelco_d_to = 0;
u8 pelco_d_c;


void UART2_config(u8 brt);   // 选择波特率, 2: 使用Timer2做波特率, 其它值: 无效.
void PrintString2(u8 *puts);
void PrintPelcoD(void);


//========================================================================
// 函数: void main(void)
// 描述: 主函数。
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void pelco_d_main(void)
{
    P_SW2 |= 0x80;//扩展寄存器(XFR)访问使能

    //P0M1 = 0x30;   P0M0 = 0x30;   //设置P0.4、P0.5为漏极开路(实验箱加了上拉电阻到3.3V)
    //P1M1 = 0x30;   P1M0 = 0x30;   //设置P1.4、P1.5为漏极开路(实验箱加了上拉电阻到3.3V)
    //P2M1 = 0x3c;   P2M0 = 0x3c;   //设置P2.2~P2.5为漏极开路(实验箱加了上拉电阻到3.3V)
    //P3M1 = 0x50;   P3M0 = 0x50;   //设置P3.4、P3.6为漏极开路(实验箱加了上拉电阻到3.3V)
    //P4M1 = 0x3c;   P4M0 = 0x3c;   //设置P4.2~P4.5为漏极开路(实验箱加了上拉电阻到3.3V)
    //P5M1 = 0x0c;   P5M0 = 0x0c;   //设置P5.2、P5.3为漏极开路(实验箱加了上拉电阻到3.3V)
    //P6M1 = 0xff;   P6M0 = 0xff;   //设置为漏极开路(实验箱加了上拉电阻到3.3V)
    //P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口
        P1M1 = 0x00;   P1M0 = 0x00;

        XOSCCR = 0xc0;//启动外部晶振
        while(!(XOSCCR & 0x01));//等待时钟稳定
        CLKDIV = 0x00;//时钟不分频
        CLKSEL = 0x01; //选择外部晶振

    UART2_config(2);    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 无效.
    EA = 1; //允许全局中断
   
    PrintString2("STC8H8K64U UART2 Test Programme!\r\n");//UART2发送一个字符串
        pelco_d_r = 0;

    while (1)
    {
      if((TX2_Cnt != RX2_Cnt) && (!B_TX2_Busy))   //收到数据, 发送空闲
      {
            //S2BUF = RX2_Buffer;
                        pelco_d_c = RX2_Buffer;
            //B_TX2_Busy = 1;
            if(++TX2_Cnt >= UART2_BUF_LENGTH)   TX2_Cnt = 0;

                        //超时,接收归零
                        //if(pelco_d_to > 100)
                        //{
                        //    pelco_d_num = 0;
                        //}
                       
                       
                        //在空闲状态下收到0xff,则进入等待状态。
                        /*if(pelco_d_num == 0)
                        {
                          if(RX2_Buffer == 0xff)
                          {
                                pelco_d_r = 0xff;
                                  pelco_d_num = 1;
                          }
                        }
                        */
                        if(pelco_d_c == 0xff)
                        {
                          pelco_d_r = 0xff;
                                pelco_d_num = 1;
                        }
                        //在等待状态下,接收到的新字节放到数组,num递增,7个后归零
                        else if(pelco_d_num > 0)
                        {
                          pelco_d_r =pelco_d_c;
                                pelco_d_num ++;
                                if(pelco_d_num > 6)
                                {
                                  pelco_d_num = 0;
                                        PrintString2("pelco_d data is:");
                                        PrintPelcoD();
                                        PrintString2("\r\n");
                                }   
                        }
                       
      }
    }
}


//========================================================================
// 函数: void PrintString2(u8 *puts)
// 描述: 串口2发送字符串函数。
// 参数: puts:字符串指针.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void PrintString2(u8 *puts)
{
    for (; *puts != 0;puts++)   //遇到停止符0结束
    {
      while(B_TX2_Busy);
                S2BUF = *puts;
      B_TX2_Busy = 1;
      while(B_TX2_Busy);
    }
}

//========================================================================
// 函数:
// 描述:
// 参数:
// 返回:
// 版本:
// 日期:
// 备注:
//========================================================================
void PrintPelcoD(void)
{
    u8 i = 0;
        for (; i<7; i++)   //遇到停止符0结束
    {
      while(B_TX2_Busy);
               
                S2BUF = '0';
      B_TX2_Busy = 1;
      while(B_TX2_Busy);
               
                S2BUF = 'x';
      B_TX2_Busy = 1;
      while(B_TX2_Busy);


                if((pelco_d_r/16) > 9)
                {
                  S2BUF = ('a' + (pelco_d_r/16) - 10);
                }
                else
                {
                  S2BUF = ('0' + (pelco_d_r/16));
                }
      B_TX2_Busy = 1;
      while(B_TX2_Busy);

                if((pelco_d_r%16) > 9)
                {
                  S2BUF = ('a' + (pelco_d_r%16) - 10);
                }
                else
                {
                  S2BUF = ('0' + (pelco_d_r%16));
                }
      B_TX2_Busy = 1;
      while(B_TX2_Busy);




                S2BUF = ' ';
      B_TX2_Busy = 1;
      while(B_TX2_Busy);
    }
}

//========================================================================
// 函数: SetTimer2Baudraye(u16 dat)
// 描述: 设置Timer2做波特率发生器。
// 参数: dat: Timer2的重装值.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void SetTimer2Baudraye(u16 dat)// 使用Timer2做波特率.
{
    AUXR &= ~(1<<4);    //Timer stop
    AUXR &= ~(1<<3);    //Timer2 set As Timer
    AUXR |=(1<<2);    //Timer2 set as 1T mode
    T2H = dat / 256;
    T2L = dat % 256;
    IE2&= ~(1<<2);    //禁止中断
    AUXR |=(1<<4);    //Timer run enable
}

//========================================================================
// 函数: void UART2_config(u8 brt)
// 描述: UART2初始化函数。
// 参数: brt: 选择波特率, 2: 使用Timer2做波特率, 其它值: 无效.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART2_config(u8 brt)    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 无效.
{
    /*********** 波特率固定使用定时器2 *****************/
    if(brt == 2)
    {
      SetTimer2Baudraye(65536UL - (MAIN_Fosc / 4) / Baudrate2);

      S2CON &= ~(1<<7);   // 8位数据, 1位起始位, 1位停止位, 无校验
      IE2   |= 1;         //允许中断
      S2CON |= (1<<4);    //允许接收
      P_SW2 &= ~0x01;
      P_SW2 |= 0;         //UART2 switch to: 0: P1.0 P1.1,1: P4.6 P4.7

      B_TX2_Busy = 0;
      TX2_Cnt = 0;
      RX2_Cnt = 0;
    }
}

//========================================================================
// 函数: void UART2_int (void) interrupt UART2_VECTOR
// 描述: UART2中断函数。
// 参数: nine.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART2_int (void) interrupt 8
{
    if((S2CON & 1) != 0)
    {
      S2CON &= ~1;    //Clear Rx flag
      RX2_Buffer = S2BUF;
      if(++RX2_Cnt >= UART2_BUF_LENGTH)   RX2_Cnt = 0;
    }

    if((S2CON & 2) != 0)
    {
      S2CON &= ~2;    //Clear Tx flag
      B_TX2_Busy = 0;
    }
}



hl1200aa 发表于 2024-3-17 10:53:13

抽空画了一个接口板


页: [1]
查看完整版本: 慢搓PELCO_D解码板