基于树莓派的慢扫描电视(SSTV)摄像机

标签:树莓派 SSTV 业余无线电 HAM

春眠 发布于 2015-03-11 14:55

2967435_orig.jpg
在本教程中,我用配有PiCam摄像头的Raspberry Pi做为无线摄像机,可以在大约一百米内传送影像。影像的传送是通过慢速扫描电视(SSTV)技术通过过业余无线电台(俗称HAM或火腿)在2米波段(144.5MHz)传输的。

作者:PA3BYA
原文链接:http://www.agri-vision.nl/CMS/index.php?option=com_content&view=article&id=54&Itemid=101

Raspberry Pi可以自行发出高频FM讯号,低功率传输时无须使用额外的电子产品。若功率须些微增加,多加上一个单晶体或双晶体的放大器即可。另外,推荐使用低通滤波器以过滤高频讯号。

此教程还包含了侦测动态物体的Python程式码,让Raspberry Pi做为远超过一般WiFi网路范围的无线监视摄像机。

请注意!原则上你必须拥有业余无线电执照才能只用此装置。

以下是装置图示,请按照步骤进行。项目程式码可以在我的部落格或我的GitHub网页找到。

特别感谢KI4MCW (SSTV), Oliver Mattos 以及 Oskar Weigl (PiFm)

图注:antenna 天线/Pi NoIR Camera PiNoIR摄像机/PiFace control & display PiFace控制显示面板/Battery 电池
便携式SSTV摄像机会拍摄影像,并且通过业余无线电台SSTV摄像机传送画面。

材料
Raspberry Pi,操作系统我选择Raspbian,因为它是Raspberry Pi最热销也有良好支援的操作系统。
Pi NoIR摄像机模组 Maker Shed网站商品编号 #MKRPI6。一般的PiCam或USB网路摄像机亦可。
Pi Face控制显示面板(非必要), 但推荐选购。我打算把Raspberry Pi当作随拍慢速扫描电视摄像机来使用,因此有显示面板跟按钮非常方便。 
电池,5伏USB移动电源
一小段导线,做天线会使用到
束带,支撑天线用
牛皮纸胶, 这是很多项目的必需品

Step 1: 连接硬件

2967435_orig.jpg 9620667_orig.jpg 9625413_orig.jpg
在这个项目里,会用到的硬件只有Raspberry Pi、Pi NoIR摄像机、PiFace控制显示面板和做为天线的一段电线。
为达到可携性,用牛皮纸胶布把一个5V USB电池组黏到Raspberry Pi外壳上。

Step 2: 拍摄画面

首先要做的是拍摄要传输的画面,用 raspistill 指令行功能就能轻松达成:

raspistill -t 1 --width 320 --height 256 -e png -o /tmp/image.png

针对SSTV,我们需要320x256象素的小影像。它会以PNG影像档格式存到 /tmp 目录。

Step 3: 将影像转换为SSTV声音文件

接着,我们要把影像转换为可以无线传输的声音文件。Raspberry Pi有一些SSTV指令可以选择。
第一个拿来测试的是PiSSTV,这是一种Python指令。它可以用,但速度非常慢,一个影像要好几分钟才能转换完成。(可以参考在我的Blog上的细节。)
接着我找到由 ham KI4MCW 罗伯特‧马歇尔(Robert Marshall)所编写的简单的C语言指令。可惜在前导音调中有一些错误,但都很容易修正。我还把它改得更有弹性,可以在指令行设定声音采样频率。
我的指令的源码可以在GitHub找到。编译源码:

pi@rpicamera ~/sstv $ sudo apt-get install libgd2-xpm-dev 
pi@rpicamera ~/sstv $ sudo apt-get install libmagic-dev 
pi@rpicamera ~/sstv $ gcc -lgd -lmagic -o pisstv pisstv.c
执行程序:

pi@rpicamera ~/pisstv $ ./pisstv /tmp/image.png 22050 
Constants check: 
rate = 22050 
BITS = 16 
VOLPCT = 20 
scale = 6553 
us/samp = 45.351474 
2p/rate = 0.000285 
Checking filetype for file [/tmp/image.png] 
File is a PNG image. 
Input file is [/tmp/image.png]. 
Output file is [/tmp/image.png.wav]. 
Writing audio data to file. 
Got a total of [2589556] samples. 
Done writing to audio file. 
Created soundfile in 4 seconds.
我们可以看到SSTV声音档只花了4秒钟就建立完成。一切都很顺利。下一步:无线传输声音

Step 4: 用PiFM传输声音

可以加装一个无线发射器,像可携式无线收发器那样,但让Raspberry Pi自己产生高频讯号有趣多了。这都要感谢Oliver Mattos和Oskar Weigl的PiFM软体(可以参考我们的Raspberry Pi)。

在这里可以找到他们的程序源码。它已经有很大的进步:最初的版本很简单,但使用了所有的CPU周期,而且讯号会受到其他程序运作时产生的假讯号干扰。最新版本使用的是DMA,运作很顺畅,也不会占用所有的CPU周期。但这个程序源码现在复杂多了。
Oliver and Oskar有很大的贡献,但PiFm软体用在HAM们的业余无线电和SSTV就不适合。主要有两个问题。首先是频宽太大,第二个是定时问题。定时对SSTV很重要,而它有些误差。

Step 5: 降低频宽

Gqrx_2_2_-_rtl_0.jpg Gqrx_2_2_-_rtl_0-2.jpg Gqrx_2_2_-_rtl_0-3.jpg
降低频宽非常简单。每位HAM都知道,频宽可以由频率调变的调变系数设定,和调变高频载体的声音讯号音量相等。在源码里,它是单一个值;可以在 Outputter/class的 consume 函数找到。
这是原来的程序源码:

void consume(float* data, int num) { 
for (int i=0; i<num;i++){ 
float value = data[i]*8; // modulation index (AKA volume!)
我做了这个值的指令行参数。新的程序源码是这样:

void consume(float* data, int num) { 
for (int i=0; i<num;i++){ 
float value = data[i]*modulation_index; // modulation index (AKA volume!) (original 8)
可惜这样效果不好,仍然有很强的边带,所以在此软体的未来版本中还需要多加关注。
第一张图是全频宽FM信号的频谱图。
第二个频谱图显示降低的频宽。调整中间的波峰后得到乾净的信号,但还需要清除边带。
最后一张图是PiFm最初版本的降低频宽信号,频宽很棒,但信号受到CPU执行其他程序时产生的干扰。

Step 6: 调整定时

multiscan_slanted.jpg multiscan_straight.jpg
https://github.com/AgriVision/pisstvPiFm的声音传输采样率稍微增加或减少时,听者几乎感觉不到差别,但对於SSTV就不一样了,SSTV的定时需要很精准。
稍有误差的采样率会造成影像倾斜,像在第一张图所看到的。
第二张图是采样正确的相同声音档。
修正定时很简单,只要修正源码中的定时常数。
//clocksPerSample = 22500.0 / rate * 1373.5; // for timing, determined by experiment clocksPerSample = 22050.0 / rate * timing_correction; // for timing, determined by experiment
这边可以看到我用变数 timing_correction 来取代定时常数(1373.5),可以由指令行来设定。个别的Raspberry Pi会有不同的数值。在我的例子里,数值是1414.0。我想知道适合你的设定值是多少,请在下面留言告诉我。关於其他程序源码的修改,请参考在GitHub的原始档案。

Step 7: 新增呼号

开始用你的业余无线电呼号授权传输SSTV信号时,需要在每次传输时传送你的呼号,所以我们要把这项资讯新增到影像里。
我们可以从指令行用 imagick或从Python影像资料库(PIL)来完成。这个项目里两种都有使用。

Step 8: 捕捉动态

现在我们可以撷取影像并用PiFm来顺利传送了,接下来我们的任务是在镜头前有动静时触发影像撷取。我把这个指令放在Python,搭配PIL。这个程序源码很简单,它会比较前一个影像和当前影像的象素。如果变化太大,就会传送影像。
这里是程序源码的片段:

<pre contenteditable="false" name="code" class="cpp"># loop forever while (True): 
# grab comparison image 
imgnew, bufnew = captureImage() 
# Count changed pixel 
changedPixels = 0 
for x in xrange(0, 320): 
for y in xrange(0, 256): 
# Just check red channel as it's dominant for PiCam NoIR 
pixdiff = abs(buf[x,y][0] - bufnew[x,y][0]) 
if pixdiff > threshold: 
changedPixels += 1 
# Transmit an image if pixels changed 
if changedPixels > sensitivity: 
# Swap comparison buffers 
img = imgnew 
buf = bufnew 
transmitImage(img.copy())</pre><br>
同样地,完整程序源码可以在GitHub页面找到。
 
Frankyang2017-08-17 08:51:19
hello  我是 BH4TIN
飞鱼BH1JSS2015-03-18 09:55:32
不错,业余无线电台玩出花样来了

作者

2粉丝 3作品 0随笔
  关注 私信

作者的最新作品


阅读数: 39229