查看原文
其他

Luat模块时钟同步的N种方法

Delectate 合宙Luat 2020-01-17

很多场景中,由于业务需要,模块需要保持正确的系统时钟,才能正常工作。但是模块上电后的初试时间戳是1338516000(即2012/06/01,10:00:00),所以同步时钟成为了开发者要解决的重要问题。

 

首先我们捋一下思路——模块要如何执行动作,执行何种动作,才可以同步时钟呢?基于这个疑问,我们先做一个定义:

只有当模块和外界以某种方式产生数据交互,且模块收到完整数据并正确解析时,才可以完成对应操作。

 

定义中指的“某种方式”是没有局限性的,也就是说开发者可以自由发挥想象力,只有你想不到,没有它做不到。接下来为大家介绍一些常见的和不常见的方式,方便开发者根据不同场景选择合适的方式。

(下文中,如无额外说明,均指Lua方式的二次开发)

 

一、常见且常用的NTP

NTP是大家众所周知的、广泛使用的、精确可靠的时钟同步方式。具有高效、准确、流量少,易实现的特点。NTP这种方式,就是使用“定义”中的蜂窝网络,和远端NTP服务器交互数据,收到服务器的回应并解析,重新设定系统时钟,完成同步。

 

我司的Lua代码也提供了相应的基础库,开发者只需要简单的调用相应的函数,即可实现同步时钟:

 

LuaScript:

requirentp--对,就这一行代码即可哦

 

LuaTask:

requirentp

ntp.timeSync()--只同步一次

--ntp.timeSync(1)--每1小时同步一次

 

实际运行一下,看看效果如何:

 

 

需要注意的是:

NTP使用的是多个免费公共的NTP服务器来同步时钟,所以不能保证任何时间任何地点都能百分百同步到正确的时间。如果开发者项目中的业务逻辑严格依赖于时钟同步功能,建议使用开发者自己的服务器进行同步。

 

二、精确到纳秒的GPS

如果开发者使用的是Air8xx系列模块,或者是其他挂载GPS模块的Air2xx系列产品,那么可以是用这个方式进行同步。

 

由于卫星定位系统的特性,每颗卫星都有精确到纳秒级的铯原子钟,才能实现全球的精确定位。故此,当开发者使用含GPS功能的模块时,可以使用GPS授时特性实现同步时钟。此种方式,就是使用“定义”中的GPS卫星信号,解析数据,设定模块系统时钟,完成同步。

 

实现原理并不复杂,只要解析GPS模块上报GGA即可:

格式:

$--GGA,hhmmss.ss,llll.ll,a,yyyyy.yy,a,x,xx,x.x,x.x,M,x.x,M,x.x,xxxx*hh

示例:

$GPGGA,065545.789,2109.9551,N,12023.4047,E,1,9,0.85,18.1,M,8.0,M,,*5E

 

名称

样例

单位

描述

消息ID

$GPGGA


GGA 协议头

UTC 时间

065545.789


hhmmss.sss

纬度

2109.9551


ddmm.mmmm

N/S

N


N=北,S=南

经度

12023.4047


dddmm.mmmm

E/W



W=西,E=东

定位指示



0:未定位

1:SPS 模式,定位有效

2:差分,SPS 模式,定位有效

3:PPS 模式,定位有效

卫星数目

9


范围 0 到 12

HDOP

0.85


水平精度

MSL 幅度

18.1


单位

M


大地

-2.2


单位

M


-

差分时间

8.0

当没有 DGPS 时,无效

差分 ID

0000



校验和

*5E



<CR><LF>



消息结束

 

当然,解析RMC亦可哦:

格式:

$--RMC,hhmmss.ss,A,llll.ll,a,yyyyy.yy,a,x.x,x.x,xxxx,x.x,a*hh

例句:

$GPRMC,100646.000,A,3109.9704,N,12123.4219,E,0.257,335.62,291216,,,A*59

 

名称

样例

单位

描述

消息ID

$GPRMC


RMC 协议头

UTC 时间

100646.000


hhmmss.sss

状态

A


A=数据有效;V=数据无效

纬度

2109.9704


ddmm.mmmm

N/S

N


N=北,S=南

经度

11123.4219


dddmm.mmmm

E/W



W=西,E=东

地面速度

0.257

Knot(节)


方位

335.62


日期

291216


ddmmyy

磁变量



-

校验和

*59



<CR><LF>



消息结束

 

我司的Lua代码也提供了相应的基础库,开发者只需要简单的调用相应的函数,即可实现同步时钟:

 

LuaScript:

require"gps"

gps.init()--初始化

gps.setfixmode(0)--GPS+BD定位

gps.settimezone(gps.GPS_BEIJING_TIME)--设置时区为北京

gps.open(gps.DEFAULT,{cause="time_sync"})--打开gps

--剩下的gps.lua都会自动处理好了

 

 

 

LuaTask:

require"gps"

requiresys

requiremisc

 

gps.open(gps.DEFAULT,{tag="time_sync"})

 

sys.timeLoopStart(function()

If gps.isFix then

misc.setClock(gps.getUtcTime())

end

end, 60000)

 

相对于使用NTP等方式,此方法不受蜂窝网络限制,即使无卡或者卡欠费,也可以准确校时。但是需要注意的是,此种方法略有局限性:

1、必须使用有GPS芯片的模块,或者挂在GPS模块,且必须可以搜到1颗以上卫星才能获取时间;

2、卫星播发的时间是UTC,所以开发者还需要根据经度判断时区做对应的加减才可以;

3、要注意定位数据的有效性。

 

三、薅基站羊毛的AT+CLTS

如果我告诉你,基站也有授时服务哦,惊不惊喜,意不意外?

 

模块首次搜到基站的时候,即可获得基站下发的时间信息。发AT+CLTS即可。

 

由于此方法通用性并不是非常强,所以仅贴出AT交互流程,有能力的开发者请自行改写lua代码吧:

 

 

需要注意的是,此方法存在一定局限性:

1、不是所有的基站都会下发,存在返回是nil的情况(具体要看基站的配置);

2、只有附着的那一刻是准确的,随后再发AT+CLTS都是恒定值(即附着时),除非进入飞行模式后再离开飞行模式,重新获取一次;

3、需要有卡才能使用。

 

四、让运营商来给你打工的SMS

有什么办法让运营商来给你打工吗?当然啦,今天就让运营商来出一次血——利用运营商下行短信的时间同步模块时钟。

 

LuaScript & LuaTask:

requiresms

 

local function procnewsms(num, data, datetime)

print("procnewsms", num, data, datetime)

end

 

sms.regnewsmscb(procnewsms)

 

sms.send("10086", "hi, china mobile")

--给10068发一条内容为“hi, china mobile”的短信,等他返回短信的时候,就自带datetime啦,开发者可以根据datetime同步时系统时钟。

 

此种方法基本上百试百灵,而且给运营商发短信时双向免费的哦;不过一定要注意使用的SIM卡是否又短信功能。如果没有,是无法使用这个方法同步时钟的。

 

结束语

不论是直接去访问time.123cha.com这种提时钟的网页,抓取关键数据;还是通过mqtt传输时间字符串;哪怕打电话给模块,通过DTMF设置时间,都是完全没问题的。

 

正如“定义”所言,只要能产生数据交互,一切就皆有可能。


    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存