基于STM32L4+BC28(全网通) NBIOT开发板原理图教程源码等开源分享 FCORE_BC28L4 NB-iot原理图、教程和源码 资料下载见附件,电脑登入 源码网盘链接:https://pan.baidu.com/s/18gkI9U1sVvauu_VzuUjauA 5 t/ r1 ^% }/ i- G4 W3 K" X# w" G0 n 提取码:6yd2 |
基于STM32L476+64M QSPI接口PSRAM(IPS6404L)开源分享(含源码)
基于STM32L4R9 的QuadSPI Flash 通讯速率不理想经验分享
STM32L4超低功耗功能概述
基于STM32L431RC Standby和RTC中断唤醒经验分享
基于STM32L431的睡眠模式经验分享
STM32L4R9 的 QuadSPI Flash 通讯速率不理想
STM32L4、STM32L4+和STM32G4系列 微控制器上的专利代码读取保护
STM32L433在STOP模式USART不能工作的解决办法
【实测教程】基于STM32L4系列的实测教程分享合集
STM32L4系列MCU的五种振荡器和使用说明
添加完宏定义后再给自己添加的xxx.c文件添加对应函数如下:
******************************驻网测试函数***************************
void _BC28_SET_(void)0 Q& t+ Q. q8 H% @3 @7 y* [5 k
{8 R' o o" {% V. I K
_DEBUGE_FLEG=0;
. N& ]9 o( O6 k1 l; I
//关闭操作标志位;
// _BC28_RESET_();
% ?8 S+ l Q1 K7 g& R
printf("AT+NRB\r\n ");
HAL_Delay(2000);
_BC28_CMD_(CMD_ECHO, DCT_OK,300,4);% x. g: h% g$ s
_BC28_CMD_(CMD_CUFN_ASK, DCT_CUFN_ASK_1,300,4);
_BC28_CMD_(CMD_CUFN_SET, DCT_OK,300,4);
_BC28_CMD_(CMD_EDRX_SET, DCT_OK,300,4);
_BC28_CMD_(CMD_NCDP, DCT_OK,1000,4);% L' v* v! b# T, Z' f. }! ?% W
_BC28_CMD_(CMD_NCDP_ASK, DCT_OK,300,4);( w( z) e" Y$ X' l
_BC28_CMD_(CMD_CGATT_SET, DCT_OK,300,2);
_BC28_SIGNALASK_();
if(_BC28_CMD_(CMD_CGATT_ASK, DCT_ADCGATT1,2000,4))
{
HAL_GPIO_WritePin(STATE_LED_GPIO_Port, STATE_LED_Pin, GPIO_PIN_SET);1 R% ]/ J/ M( m0 B& I* ?1 {
}
_BC28_CMD_(CMD_CGPADDR, DCT_OK,300,4);
_BC28_CMD_(CMD_NNMI_SET, DCT_OK,300,4);
_BC28_CMD_(CMD_NSMI_SET, DCT_OK,300,4);
_BC28_CMD_(CMD_NNMI_ASK, DCT_MNMI_1,300,4);
_BC28_CMD_(CMD_NSMI_ASK, DCT_NSMI_1,300,4);
HAL_Delay(200);* `& h! k# S7 z7 E2 u1 A
if(_BC28_CMD_1(CMD_send_TexT, DCT_RCVE,600,3,4))
{9 `1 o4 X2 S. D3 b/ L3 L
_BC28_SET_();+ U+ h# `6 x- a8 h6 r
}: w, D4 f* U' a4 m0 V. e1 Q
_BC28_CMD_(CMD_AT, DCT_OK,300,20);3 b* _' S9 q' c/ P4 H& i5 z
HAL_GPIO_WritePin(STATE_LED_GPIO_Port,STATE_LED_Pin,GPIO_PIN_RESET); ) u( q$ t" z4 S; m1 h9 a
_DEBUGE_FLEG=1;, t4 H! I" V1 E; I; n; b. B
}' T: V. C9 Y \' V
这个测试函数的基本操作首先使用发送AT+NRB对模块进行重启,重启的过程大概需要2s左右,所以这里需要做相应的延时,防止后面命令发出去后被忽略,重启后使用char _BC28_CMD_(char* pstr_CMD,char* pstr_DCT,int delay_us,int MAXTRY)发送对应命令进行设置,从上图中可以看到基本都是最这个函数的调用,而这个函数实际是将串口的发送函数进行封装了,里面添加了延时,发送次数等参数的设置,熟悉C编程的朋友看到这里也就明白了,我们 实际也是使用printf函数进行命令的发送,这里涉及到对串口发送函数的重定向。发送函数的重定向可以使用直接用寄存器进行,memset()和strstr()函数需要包含string.h头文件,由于用到printf函数所以还需要xxx.h文件开头添加如下是代码:6 I* M, u( f* u ?1 U' |
xxx.c文件中添加如下代码7 ~, z' F9 ? Z- I( C( w/ L
***************************xxx.c中命令发送函数**********************
char _BC28_CMD_1(char* pstr_CMD,char* pstr_DCT,int delay_us,int MAXTRY,int MAXWAT)
{, U8 X9 \5 U0 a6 Z
char *strx;% \4 M- V' @( q# y0 _% u
int i=0,j=0;
memset(RX_BUFFER_lp, NULL, 200);4 R; `5 q* c; R5 v0 N6 O8 D8 Q
strx=NULL;
while(strx==NULL && j < MAXTRY); y; B+ J. J" }# O7 L$ K
{6 ^9 m: ]4 ^7 T9 E
memset(RX_BUFFER_lp, NULL, 50);& r7 l8 o w- |2 k
HAL_Delay(50);
printf("%s",pstr_CMD);
j++;
i=0;
while(strx==NULL && i < MAXWAT)# k* X: r, u; o* V
{: a [7 N$ Y) l
strx=strstr((const char*)RX_BUFFER_lp,(const char*)pstr_DCT);! K C) \7 g! F4 G9 t1 @
HAL_Delay(delay_us);
i++;
}; E( x% t0 I, Z# T
}& @2 ]0 o/ X3 `$ N8 B0 s
if(strx==NULL )& E9 Q: g6 c2 {* ]6 s) z
{- U: m' @8 V- R! g" d, q V: Y
return 0;
}: e7 |) P6 f) R- o, D. a' O- t
else6 V' j; e4 e4 ]- y( c' ]
{
return 1;
}
}
char _BC28_CMD_(char* pstr_CMD,char* pstr_DCT,int delay_us,int MAXTRY)
{
char *strx;3 k J, p n1 s' Y5 [
int i=0;6 ^ P* {' F( K+ F: O% V
memset(RX_BUFFER_lp, NULL, 200);" E/ \3 _8 A' M
strx=NULL;
HAL_Delay(50);
while(strx==NULL && i < MAXTRY)- w) a# @' c1 D/ [" ?0 c4 G
{* M# v' I1 j5 X: w8 V9 P
memset(RX_BUFFER_lp, NULL, 200);: q5 Z9 t9 \5 _* @+ N. C8 {
printf("%s",pstr_CMD); //???lpuart????????
HAL_Delay(delay_us);
strx=strstr((const char*)RX_BUFFER_lp,(const char*)pstr_DCT);4 u7 P; Z* y1 M) J, _& j
i++;: s" \, E9 r5 j! u/ m# f
}
if(strx==NULL )
{
return 0;
}3 f4 r; y- w: l( p, F: ]" ?
else
{
return 1;
}
}
char _BC28_SIGNALASK_(void)( c1 X0 r$ D x
{
memset(RX_BUFFER_lp, NULL, 200);) d4 D, P# A& U5 P2 f. W) @
char CSQ=99;- e6 B0 i) s5 d) u r; w p
char *strx;
int i=0,j=0;
strx=NULL;9 O& ~/ s* b' E# ]- E c9 [% d$ C
HAL_Delay(50);) |8 L: y. x: r' A
while(strx==NULL && i < 30 )
{3 v J" E6 z9 Y4 P- c
printf("%s",CMD_CSQ); & k; `9 G* r2 y. O# ]
HAL_Delay(200);
strx=strstr((const char*)RX_BUFFER_lp,(const char*)DCT_ADCSQ);
i++;
if(strx)4 \! m8 o+ z( X6 u% A
{
CSQ=(strx[5]-0x30)*10+(strx[6]-0x30);# k9 |; c5 f9 t- e
if(CSQ==99)$ P' v( c6 @- }7 y) A8 ]
{9 a, s: V; j! O( _7 e
memset(RX_BUFFER_lp, NULL, 50);4 I( e9 k9 [8 t% M& V5 O7 c
while(CSQ==99 && j < 300)" t- i R0 i6 t% }, h# ^ _
{
printf("%s",CMD_CSQ); ֵ
HAL_Delay(400);
strx=strstr((const char*)RX_BUFFER_lp,(const char*)DCT_ADCSQ);
CSQ=(strx[5]-0x30)*10+(strx[6]-0x30);! H* m9 M* q2 U
j++;
}
}* m% J/ d4 ~2 H7 `
}4 } R: l4 T9 h
}* `# k% h a+ S+ W% q0 W. t& p
if(CSQ==99)- s6 o. v! n$ e
{
return 0;
}
else
{& b2 U2 q9 V1 P0 ~0 I# W/ _4 F6 _
return CSQ;
}
}
char _BC28_SIGNALASK_(void)% F) x/ J: c6 {$ F
{
memset(RX_BUFFER_lp, NULL, 200);
char CSQ=99;) N% P! Q+ M/ J3 F
char *strx;
int i=0,j=0;: h: `5 J3 \- z6 v, p) ?2 B8 g
strx=NULL;
HAL_Delay(50);4 n z" P/ D8 d n/ ^( x
while(strx==NULL && i < 30 )
{. m1 }2 }" f0 K) h
printf("%s",CMD_CSQ); //查看获取CSQ值/ r* X0 ?! N7 @+ f3 n0 n
HAL_Delay(200);" T' d+ S% B ?
strx=strstr((const char*)RX_BUFFER_lp,(const char*)DCT_ADCSQ);
i++;
if(strx)+ u# Y1 Q0 b9 Z
{
CSQ=(strx[5]-0x30)*10+(strx[6]-0x30);//信号转码
if(CSQ==99)
{
memset(RX_BUFFER_lp, NULL, 50);5 u+ q5 `8 j% R2 s0 O0 l
while(CSQ==99 && j < 300)/ I3 f1 M+ j* X" a4 O0 x. O
{
printf("%s",CMD_CSQ); //查看获取CSQ值9 P( t2 F. k7 ~! [7 i
HAL_Delay(400);4 f% C& z: B& g; C2 Z+ n9 C! `; q, H
strx=strstr((const char*)RX_BUFFER_lp,(const char*)DCT_ADCSQ);
CSQ=(strx[5]-0x30)*10+(strx[6]-0x30);//信号转码/ H$ X3 c0 @7 _1 d! v; p+ J/ M
j++;4 o: Q1 v6 ]1 {
}
}
}
}1 L- s9 X6 Q: \' y& ~" Y) q! V
if(CSQ==99)5 j ~; w K$ ]& w4 U( p( ?& f7 _
{1 n: D* P! U' g& _3 F) ^
return 0;! X: Z# c& |2 }/ b. H; F* j
}
else5 ?( B' O0 x+ L: g; x
{8 N6 R) O6 ?6 h+ g5 j; A/ L
return CSQ;% O! u3 O: h4 u T4 g' t2 j% ]
}
}) k* ?4 j* e' k T
******************************命令宏定义***************************** Z) L& a- _. _# _7 B$ ~
#define CMD_AT_RESET "AT+NRB\r\n"/ x0 \8 | e* t* Z7 e( r. I9 T
#define CMD_AT "AT\r\n") T" M( _( b( |; R) H
#define CMD_ATI "ATI\r\n"" E! Q2 L- s, p: _% `: G9 z
#define CMD_CUFN_RESET "AT+CFUN=0\r\n"% t. P1 q6 w' U1 B! X2 b: A. P
#define CMD_CUFN_ASK "AT+CFUN?\r\n"1 B9 J- f9 e2 t N
#define CMD_CUFN_SET "AT+CFUN=1\r\n"
#define CMD_NTSETID "AT+NTSETID=1,863703030570134\r\n"
#define CMD_NCDP "AT+NCDP=49.4.85.232,5683\r\n"//"AT+NCDP=117.60.157.137,5683\r\n"
#define CMD_NCDP_ASK "AT+NCDP?\r\n"
#define CMD_CGPADDR "AT+CGPADDR\r\n"- j6 N0 L3 d6 ]0 s
#define CMD_SINGNALASK "AT+NUESTATS\r\n"
#define CMD_ICID "AT+CIMI\r\n"
#define CMD_ECHO "ATE\r\n"" V1 t) S2 ~8 j; I
" H" Z1 g" s( b& X5 c- Z5 ^5 P
#define CMD_DUTONET "AT+NCONFIG=AUTOCONNECT,TRUE\r\n"0 h+ I: s& [" h, n% T/ G/ t. r
#define CMD_CGDCONT "AT+CGDCONT=1,\"IP\",\"psm0.edrx0.ctnb\"\r\n"9 W1 L( i8 Q9 q" k: G7 Y
#define CMD_CGDCONT_ASK "AT+CGDCONT?\r\n"
#define CMD_NNMI_SET "AT+NNMI=1\r\n"9 u: {& j! G( L, a! _
#define CMD_NSMI_SET "AT+NSMI=1\r\n"
#define CMD_NNMI_ASK "AT+NNMI?\r\n". X8 P# ^) L2 \0 L! h
#define CMD_NSMI_ASK "AT+NSMI?\r\n"1 Y" b7 H& k7 C+ B ^' K
" `) M% `' m* _7 v, p
#define CMD_SEND_DAT "AT+NMGS=18,980012010001123456785A2608CE00A3E1FF\r\n"# |7 s9 z/ b Q2 g! x1 T3 I {3 T
#define CMD_NBAND_ASK "AT+NBAND?\r\n"& J4 q% o' \( v0 |3 o1 k
#define CMD_NBAND_SET "AT+NBAND=5\r\n"
#define CMD_CIMI "AT+CIMI\r\n". ?( k0 H& U: j9 E
#define CMD_CGATT_SET "AT+CGATT=1\r\n"
#define CMD_CGATT_RESET "AT+CGATT=0\r\n"
#define CMD_CGATT_ASK "AT+CGATT?\r\n"2 o$ ]3 d0 O( e
#define CMD_CSQ "AT+CSQ\r\n"
#define CMD_CEREG_ASK "AT+CEREG?\r\n"8 s! C9 d7 v+ S, `
#define CMD_CEREG_SET "AT+CEREG=1\r\n"
#define CMD_CSCON_ASK "AT+CSCON?\r\n"
#define CMD_CSCON_SET "AT+CSCON=1\r\n"0 S- ^- j& T2 ?: s% t- @
#define CMD_PSM_SET "AT+CPSMS=2\r\n"* t# {& ^* g/ X9 e4 V! R- v
#define CMD_EDRX_SET "AT+CEDRXS=3\r\n"
#define CMD_send_TexT "AT+NMGS=21,9800170100010000000100000001FFFF000100FF47\r\n"
#define DCT_SIGNALASK "Signal power"2 ]2 O& [* T8 T6 m
#define DCT_RCVE "+NNMI:18,"
#define DCT_RST "Neul"
#define DCT_OK "OK"
#define DCT_MNMI_1 "+NNMI:1"
#define DCT_NSMI_1 "+NSMI:1"
#define DCT_CUFN_ASK_1 "+CFUN:1"1 ~0 E5 l# m9 V$ ]( g& [3 b
#define DCT_CUFN_ASK_0 "+CFUN:0"
#define DCT_CGDCONT "+CONT:1"4 H9 C" X, G6 t5 w, p3 Y' A- }5 P: W
#define DCT_BAND5 "+NBAND:5"! E9 c4 ~& X! T1 M5 [9 k# i0 `$ S
#define DCT_ADCGATT1 "+CGATT:1" C: e, f6 ~' B+ W" y' ?
#define DCT_ADCGATT0 "+CGATT:0"
#define DCT_ADCSQ "+CSQ"+ H/ U# |- \, O! ?: t. p; \+ l1 Y
#define DCT_CSCON1 "+CSCON:0,1") R3 g/ l2 i7 B$ k# [3 w
#define DCT_CSCON0 "+CSCON:0,0"
#define DCT_NSONMI "+NSONMI:"
#define DCT_CEREG1 "+CEREG:0,1"% |- \1 Z! s8 U) a* y1 r) ^
#define DCT_CEREG0 "+CEREG:0,1"
#define DCT_SOCKET_OFF "AT+NSOCL=0\r\n"
#define DATA_IP "AT+NSOST=0,120.24.184.124,8010,%c,%s\r\n",'8',"727394ACB8221234") Y I9 E% T' Q0 B: v
#define DATA_SOCKET "AT+NSOCR=DGRAM,17,3568,1\r\n", o! p7 ]" q. H1 c$ }$ q
#define DATA_IP "AT+NSOST=0,120.24.184.124,8010,%c,%s\r\n",'8',"727394ACB8221234"; \% V% P0 y2 k3 R2 |8 e, }
#define DATA_SD_IP "AT+NSOST=0,120.24.184.124,8010"
#define DATA_SD_APN "AT+CGDCONT=1,\042IP\042,\042HUAWEI.COM\042\r\n"$ @# f9 h/ r# G7 v0 Y
到这里我们的板卡联网就成功了,接下来就是将设备连接到自己开发的云平台了,我们这里只对华为云平台进行介绍。最开始做的当然是注册和实名认证,如果是个人学习使用建议使用个人认证,送的发布信息等足够个人用户使用。在认证完后再进行下一步操作。
认证完后,在左上角产品中找到Iot物联网,点击后在右侧子菜单左上角的物联网平台(云)栏找到设备接入并点击,然后就能看到图14界面,点击立即使用;
0 V+ |( f, ]; Z
图14) P& Y( ~8 i. G7 _) R
进入后在左侧可以看到六个板块,这里点击最下面的开发中心;进入开发中心后,找到如图15板块,点击新建项目;
图15
项目名称自己随便填,符合华为云命名规则就好,所属行业我们选NB-Iot,因为我们的板卡是只支持窄带宽(NB)的,这里还要说明的是选择SIM卡的时候要注意选用NB的专用卡(即窄带宽专用卡,可以去http://www.360wulian.net/上面免费申请试用),否则是识别只能读取到对应的SIM卡ID,但是不能联网而且没有信号,如图16;$ M; S' }- d2 @/ x+ }
图16
项目建好后点击项目右侧进入如图 然后我们就进入了华为OceanConnect开发中心了如图17;
6 r0 R$ T# E. A! ~
图17% m$ K: n1 L( v5 m: n
然后点击左侧开发中心进行创建产品,可以看到如图18界面;点击自定义产品,后再弹窗口填写产品相关信息设备名字、型号、设备类型所属行业随便写应该是没问题的,主要是协议,协议按照模块支持进行选取就好,我这里选择的是CoAP,设备图片的话可以选个自己喜欢的图片传上去,嘿嘿;点击创建,这样你的产品就创建好了,如图19;# t) L% `# H' ?2 |
图18
图19- z! D$ F) {4 v* x' o" Q+ }! ?' D
额,这两张图有点密集了,大家凑合这看看就好;接下来进入的界面是云端测试的相关事项了,首先是Profile的定义,如图20,点击右侧新建服务;& N5 P* } y( m! p* N+ {
图20 J% e6 O( Z1 e* Q6 F
然后依次设置服务名称和对应属性和命令如图21所示(图被压缩的有点丑);
图21, ^' \6 i9 B2 X
属性的设置就这样了,然后设置命令,第一个open命令我设置的值是只能发0和1,用于打开和关闭led,如图22
图22
Profile的部分到这儿就结束了,翻到页头,点击图编解码插件开发,如图图23,点击新增消息在弹出窗口添加数据上报填写消息名后点击添加字段再弹出窗口填写对应信息;如图24。" B9 d- `; C& o
! l" i8 {: K. J' o) E" p$ N0 ]
图23
这里的话按照我们设备的属性,有两个状态,所以我们添加两个状态字段;然后添加的下发的命令,实际操作和上报差不多,就不演示了,最后需要做的就是部署插件了点击右上角蓝色的部署两个字就好了! w9 O6 N" g6 ^- ?: O* [) R
图24
设置完后点击右侧设备模型并将属性下的各个模块拖到左侧白色区域就好了,不拖过来的话是不能够上报和下发命令的哦,然后点击右侧上方的蓝色部署两个字等待部署好就行,如图25。* ?, h) h+ o! ?& l# ], y7 P
/ Y1 g& Q; Y% }
图250 P. i. Y. u% S) f
然后我们跳过第三步,直接进行第四步在线调试。在进入界面后点击屏幕中间的新增测试设备在弹出窗口可以看到可以添加虚拟设备和真实谁被两个选项,我们选择添加真实设备,设备名称随便写,设备标识我们填写通过AT+CGSN=1命令调出的编号,点击创建就好,如图26:
% Q8 a/ X9 S# f* i. D5 [8 U
图26
这样我们先回到云平台,左侧菜单栏点击对接信息,然后对模块进行联网,拷贝图26红色框内容;7 n j+ F" S# X, c, p" y4 W5 T+ q
3 T% g1 q) ]$ d [
图27
在串口通信助手发送AT+NCDP= <IP>,<端口>命令,返回OK后,点击左侧产品开发,在以下界面点击图28红色框区域(我们创建的设备)就可以看到我们的产品已经上线了;这样的情况就是我们的板卡已经练到云端了。如图28;
图289 n% f4 k | \1 t7 C- X* @& `- v$ _
图29
进入图29后点击右侧红色框区域,便可以调试我们的产品了。进入调试后就需要把我们的设备连接到云平台,并进行数据的上报和下发。如图30为调试界面。% j6 _3 p6 @" Y
图306 x7 f# J2 h; ?6 T
现在我们要做的事是将板卡链接到云平台,这里需要上报信息的命令AT+NMGS=<长度>,<数据>以这个格式发送信息,返回OK后,在云平台右侧效力跟踪里就能看到跟踪信息并且在应用模拟器里看到我们上报的信息如图31和图32.
图31 图32
这样我们的上报数据就正常了;如图33和图34为发送命令,图33红色发送数值为4,图34中收到的为04,这样下发命令就成功了。
" x: K1 e7 {3 B! u. o
图33
7 t; s' N/ { p& N- z, j% N
图34
2 R* C$ w- Y2 @# ]! P4 w9 k. u8 n' M% l