It's time! The camera is not yet working. Neither is the 3G modem. We're still running on the Panda Board though and it has got Wifi. So we let the boat connect to a wifi provided by Kristians android. The GPS and servo controller are fully operational and the control interface for the iPad is close to perfect. Here we go!
Raspberry Boat
Blog for the Raspberry Boat project. We're building a boat controlled by a Raspberry Pi.
Monday, June 11, 2012
Tuesday, June 5, 2012
We have 3G modem
My dad just decided to upgrade to 4G. 4 is better than 3. We all know that. So I got his old 3G modem :)
So this is where I tell the story of the 3G. Again - this is back dated so this is a long story which in reality began a few weeks after we got the modem... and ended like a week ago (september 3:rd).
A brand old Huawei E1820 |
The story of the 3G modem
You would think by now we would've learned that nothing is as easy as it seems. But we hadn't. Actually, we still haven't. We still approach every challenge with the same positive, wreckless attitude. Anyway; we had the modem and started googling for information on how to use it. It turns out it wasn't that hard. Pretty soon we found a promising looking script for wvdial.
There is no wvdial
Apparently there is no wvdial for Debian squeeze on arm... go figure! So we resorted to pppd. How hard could that be. As it turned out: very. It actually took us a whole night just to figure out that our problem was the PIN lock. Then we realized my dad didn't have the PIN code. He had the PUK code though... and an hour later we had figured out how to lock the phone using erroneous PINS so that it would respond to the PUK. Aaaaahhhh those AT codes - it brings back the old times of configuring that noisy US Robotics :)
So anyway... We unlocked the modem and then went back to googling for settings. What we wanted was, specifically, to get a Huawei E1820 to connect to Telenor and get an IP address. We tried EVERYTHING! Nothing worked. We permutated the order. Didn't work. And then... by pure f**ing luck - we found the correct configuration. So we built a start-up script to make the boat connect, get an IP address and ping a Heroku app so that we could see the IP from the iPad. After reboot it worked... almost. So we rebooted again. Nothing. A black screen.
There is no partition
Yep... after trying to get the Panda to wake up for about 20 minutes, we yanked the SD card and put it into a computer. And it contained: 16 GB unpartitioned space. Thank you Televerket!
Now, all the code for the GPS and the modules and pretty much everything else has been on Github the whole time. But not the start-up script. And not the modem settings. You know; the ones we finally got right by pure luck.
There is video
So we do a bit copy of the memory card and start searching for the word telenor in a Hex editor. Gigabyte after weary gigabyte is copied and searched... but no telenor. So we start talking about how there could be so much crap on the SD card. And then we give thanks to Sebastian for cramming the Panda with music videos. (he may argue that the extensive logging was just as bad - especially my Tourettes like rage quits like `sudo halt -fitta`).
And so finally we find it. The config is saved!
There is Raspberry
The wiped SD card put us off a bit. And by that time, we had no less than three shiny Raspberry Pi:s waiting for us. It's true - they're not as powerful - but they're damn sexy! And Raspbian - essentially Debian Wheezy - has got wvdial... oh the joy! So we pull out the old config and run it on the RPi... and NOTHING WORKS!!!
There is also keyboard
After about an hour of bug hunting, we find a mention of certain keyboards derping other USB devices. Sebastian leaves the room and ssh:s into the RPi. GREAT SUCCESS! It works. Screw you keyboard!
Monday, May 28, 2012
Eyes and nervous system
The USB hub, the cameras and the GPS have been stripped of unnecessary plastic and attached to the hull. And they actually work... sort of... except for the cameras.
Since all these posts are back dated... I'll compress the story of the cameras a bit in this post.
At the time of these photos, we started playing with video. We were done last monday - on september 3:rd!
Since all these posts are back dated... I'll compress the story of the cameras a bit in this post.
At the time of these photos, we started playing with video. We were done last monday - on september 3:rd!
First idea: VLC
Let's just install it and tell it to broadcast a stream. The stream format of choice - off course; HTTP Live Streaming. Hey! It'll work on the iPad.Fail
VLC is just to slow. There's a lag of about a second just from camera to local screen.
Second idea: ffmpeg with custom segmenter for HLS
The only official segmenter for creating HLS only works on OS X. But there is an OSS alternative. It just refuses to compile on the Panda. We spent the whole night fiddling with header files and wreaking all kinds of havoc to the Debian install on the Panda... but we got it working! That's when we realized that HLS SUCKS!
HTTP Live Streaming is the only format of live video that is supported in HTML on iOS devices. It works like this:
- You record the stream and pipe it into a
- Segmenter chops the video into little bits and shoves them into a folder. It then updates a
- m3u file. Yes you got it! It updates a playlist. So we serve the files and the playlist through
- nodejs static folder. Now all we have to do is place a video tag on the page and we're done.
Fail
Well... apart from the CPU running at about 90% and the Panda almost catching on fire, we have a lag of about 30 seconds. By tweaking the settings for ffmpeg and the segmenter, we manage to get it down to about 15 seconds. That's BAD NEWS when you're using the video to control a boat.
Third idea: ffmpeg to produce a stream of jpegs
We don't really care about framerate do we? We care about lag! If a rock is coming at you, you don't give a crap if it's choppy as long as you get the information in time so you can evade it.
So let's just let ffmpeg produce a series of jpegs and tell the browser to update regularly!
Fail
This is how browsers work when an image changes: they replace it on screen and keep the old ones in memory. That's good for optimization but CRAP for running a boat for half an hour with image updates 5 times a second.
Fourth idea: multipart/x-mixed-replace
At Devsum in Stockholm, I start talking to Eric Lawrence, author of Fiddler. He mentions the multipart/x-mixed-replace protocol which emulates server push. It doesn't work on Internet Explorer but who cares? We try it by building a node module which spawns an ffmpeg child process, captures an image and writes it to the response... and it works!
Fail
When I say works, I mean it works but it's sloooow. After thinking about it for a while, we realize what we're actually doing is this:
- Spawn a child process staring ffmpeg
- Wait for it to connect to the camera and return a jpeg
- Let the process die
- Repeat
This just isn't going to cut it.
Fifth idea: build a node module which parses a mjpeg stream
What we really want to do is keep the camera streaming and just push jpegs to the response. So we use ffmpeg to start a mjpeg stream and parse the contents of stdout to cut it up into individual jpegs.
Fail
Again it works but it's to slow. And the CPU is running at about 70%.
Interlude: 4008
It's about this time we feel life is just to easy. Let's make it a bit more interesting! Let's connect the second webcam! Say hello to our new friend: Error 4008. It turns out the webcams are bandwidth hogs. With twocameras running, we just deplete the memory of the USB. By this time, we've switched to the Raspberry Pi (you can read about that in later posts). We have currently updated Raspbian twice AND upgraded the firmware but 4008 still haunts us.
Sixth idea: let's go native!
So this is where Kristian decides javascript sucks and reverts to c. He builds a node module which cats the stream from the camera and emits events when a jpeg is ready to deliver.
Fail
Threading, threading, threading! While the code works, it just locks the process. When we start the camera, everything just dies.
Seventh idea: let's go native AND not be idiots!
The c lib gets rewritten. This time, we pass it a writeable stream (ie the response). The module spawns a child process which cats the stream, finds a jpeg, writes it to the stream and then sleeps for 1M/framerate microseconds.
Win
It works. It uses about 3% CPU. It pumps jpegs like there's no tomorrow. Still 4008 with two cameras though... but we've found a solution for that to: I taped shut the hole in the boat for the back cam.
Pandaboard and stripped down (massive) USB hub |
GPS and forward camera |
Tuesday, May 8, 2012
Monday, April 2, 2012
Control? Power!
Ok... so we have a boat and a servo controller. Now for some node. Kristian is busy writing a c lib for node to allow him to controll the servo. Meanwhile Sebastian is in charge of power.
The boat is now connected to a USB charger and it works... well the rudder does - not the engine. It's just not satisfied with 5.5 V.
Hacking c and hating mac. Kristian and macs just don't play nice.
Wednesday, March 28, 2012
We have boat
So here's the idea: We buy a radio controlled boat and rip out the reciever. We then install a servo controller instead and hook it up to a Raspberry Pi. Then we attach a GPS, two cameras - one for forward and one for backward view, and a 3G modem.
Then we install nodejs and program it to control all the devices. It will then serve up a web page which we can surf to with an iPad. The web page will display the GPS position, using Google maps, the cameras and provide a touch interface to controll the throttle and rudder.
Sound ok? Yay - let's go!
Kristian and Sebastian doing some soldering
The servo controller in place
The boat
Actual working GPS code
Aaaaand some more soldering
Subscribe to:
Posts (Atom)