Working on a fun project to track cats internally around houses. I need to track 4 individual cats around a standard 3 bedroom house using an array of sensors around the house.
I am using the camera from a wiimote, to detect the 4 brightest IR signatures in the field of view (Camera attached to ceilings looking down). Each IR blob is returned with a X and Y co-ordinate from the camera.
The cats wear a IR emitting collar and to identify which cat is which I have each collar flashing at a set frequency. 10hz, 20hz, 40hz and 80hz.
Each time a blob comes into view i record into into FIFO array with its x/y position and timestamp. When each blob comes in i look through that array bottom up and look for a recording that is in a similar position, i then calculate the delta time between those two recordings to workout which cat that is.
Obviously in loops where the IR light is still on the delta time is just the time of each loop cycle, but as these are not in the range i am looking for it doesn't seem to be an issue.
The camera outputs data at around 200hz.
While this is working the timing can be a little off at times and sometimes get some readings where cat 3 shows as 4 etc. I'm just wondering if any of you chaps have any ideas on how I might be able to achieve the above more efficiently, and with perhaps a bit more error detection. My shift array left function is also a little crude.
Code below, any feedback welcome.
Cheers
B
Code: Select all
Option ConsoleSpeed 115200
Option SignOn Off
'STACKS
public ComTaskStack(1 to 200) as byte
'I2C
PUBLIC i2c_Result as INTEGER
PUBLIC i2c_Result_bol as BOOLEAN
PUBLIC i2c_OutData(1 to 16) as BYTE
PUBLIC i2c_InData(1 to 16) as BYTE
'BLOB PROCESSING
PUBLIC blob_i as Byte
PUBLIC blob_x(1 to 4) as Integer
PUBLIC blob_y(1 to 4) as Integer
PUBLIC blob_extra(1 to 4) as Integer
PUBLIC blob_size(1 to 4) as Integer
PUBLIC timenow as Single
PUBLIC blob_position_x(1 To 20) as Integer
PUBLIC blob_position_y(1 To 20) as Integer
PUBLIC blob_lastseen(1 To 20) as Single
PUBLIC i as Byte
PUBLIC o as Byte
'CAT DATA
PUBLIC cat_i as Byte
PUBLIC cat_visible(1 to 4) as Boolean
PUBLIC cat_x(1 to 4) as Integer
PUBLIC cat_y(1 to 4) as Integer
PUBLIC cat_freq(1 to 4) as Single
PUBLIC cat_lastseen(1 to 4) as Single
'RANDOM
PUBLIC teststring as BoundedString(255)
'SETTINGS
PUBLIC movementthreshold as Integer = 40
Sub Main()
Call putpin(25,0) 'led on
CallTask "Comms", ComTaskStack
call OpenI2C(0, 11, 12, 10)
'CONFIGURE CAMERA
i2c_OutData(1) = &H30
i2c_OutData(2) = &H01
i2c_Result = I2CCmd(0, &HB0, 2, i2c_OutData, 0, i2c_InData)
Call delay(0.1)
i2c_OutData(1) = &H00
i2c_OutData(2) = &H00
i2c_OutData(3) = &H00
i2c_OutData(4) = &H00
i2c_OutData(5) = &H00
i2c_OutData(6) = &H00
i2c_OutData(7) = &H00
i2c_OutData(8) = &H90
i2c_Result = I2CCmd(0, &HB0, 8, i2c_OutData, 0, i2c_InData)
Call delay(0.1)
i2c_OutData(1) = &H07
i2c_OutData(2) = &H00
i2c_OutData(3) = &H41
i2c_Result = I2CCmd(0, &HB0, 3, i2c_OutData, 0, i2c_InData)
Call delay(0.1)
i2c_OutData(1) = &H1A
i2c_OutData(2) = &H40
i2c_OutData(3) = &H00
i2c_Result = I2CCmd(0, &HB0, 3, i2c_OutData, 0, i2c_InData)
Call delay(0.1)
i2c_OutData(1) = &H33
i2c_OutData(2) = &H03
i2c_Result = I2CCmd(0, &HB0, 2, i2c_OutData, 0, i2c_InData)
Call delay(0.1)
i2c_OutData(1) = &H30
i2c_OutData(2) = &H08
i2c_Result = I2CCmd(0, &HB0, 2, i2c_OutData, 0, i2c_InData)
Call delay(0.1)
i2c_OutData(1) = &H37
i2c_Result = I2CCmd(0, &HB0, 1, i2c_OutData, 0, i2c_InData)
Call delay(0.1)
do
'GET CAMERA DATA
i2c_OutData(1) = &H37
i2c_Result = I2CCmd(0, &HB0, 1, i2c_OutData, 0, i2c_InData)
i2c_Result = I2CCmd(0, &HB1, 0, i2c_OutData,12, i2c_InData)
blob_x(1) = cint(i2c_InData(1))
blob_y(1) = cint(i2c_InData(2))
blob_extra(1) = cint(i2c_InData(3))
blob_x(1) = blob_x(1) + Shl(blob_extra(1) And &H30, 4)
blob_y(1) = blob_y(1) + Shl(blob_extra(1) And &Hc0, 2)
blob_x(2) = cint(i2c_InData(4))
blob_y(2) = cint(i2c_InData(5))
blob_extra(2) = cint(i2c_InData(6))
blob_x(2) = blob_x(2) + Shl(blob_extra(2) And &H30, 4)
blob_y(2) = blob_y(2) + Shl(blob_extra(2) And &Hc0, 2)
blob_x(3) = cint(i2c_InData(7))
blob_y(3) = cint(i2c_InData(8))
blob_extra(3) = cint(i2c_InData(9))
blob_x(3) = blob_x(3) + Shl(blob_extra(3) And &H30, 4)
blob_y(3) = blob_y(3) + Shl(blob_extra(3) And &Hc0, 2)
blob_x(4) = cint(i2c_InData(10))
blob_y(4) = cint(i2c_InData(11))
blob_extra(4) = cint(i2c_InData(12))
blob_x(4) = blob_x(4) + Shl(blob_extra(4) And &H30, 4)
blob_y(4) = blob_y(4) + Shl(blob_extra(4) And &Hc0, 2)
timenow = Timer()
For blob_i = 1 To 4
if (blob_x(blob_i) <> 1023) AND (blob_y(blob_i) <> 1023) then
Dim match as Byte = 0
For i = 1 To UBound(blob_position_x)
o = (UBound(blob_position_x) - (i-1))
if match = 0 Then
if nearby(blob_x(blob_i),blob_position_x(o),blob_y(blob_i),blob_position_y(o),movementthreshold) = True then
match = o
end if
end if
Next i
Dim frequency as Single
if match <> 0 then
blob_position_x(match) = blob_x(blob_i)
blob_position_y(match) = blob_y(blob_i)
frequency = (timenow - blob_lastseen(match))
if frequency > 0.0075 And frequency < 0.0175 then '0.0125
cat_visible(1) = True
cat_x(1) = blob_position_x(match)
cat_y(1) = blob_position_y(match)
cat_lastseen(1) = timenow
end if
if frequency > 0.020 And frequency < 0.030 then '0.025
cat_visible(2) = True
cat_x(2) = blob_position_x(match)
cat_y(2) = blob_position_y(match)
cat_lastseen(2) = timenow
end if
if frequency > 0.045 And frequency < 0.055 then '0.050
cat_visible(3) = True
cat_x(3) = blob_position_x(match)
cat_y(3) = blob_position_y(match)
cat_lastseen(3) = timenow
end if
if frequency > 0.095 And frequency < 0.105 then '0.0100
cat_visible(4) = True
cat_x(4) = blob_position_x(match)
cat_y(4) = blob_position_y(match)
cat_lastseen(4) = timenow
end if
blob_lastseen(match) = timenow
else
Call shiftarrayleft()
blob_position_x(20) = blob_x(blob_i)
blob_position_y(20) = blob_y(blob_i)
end if
end if
Next blob_i
'timeout cat visible readings
For cat_i = 1 To 4
if (timenow - cat_lastseen(cat_i)) > 0.2 then
cat_visible(cat_i) = False
cat_x(cat_i) = 0
end if
Next cat_i
'do we want to smooth the x and y data? and lowpass filter it?
loop
End Sub
Public sub shiftarrayleft()
'shift arrays left FIFO
blob_position_x(1) = blob_position_x(2)
blob_position_x(2) = blob_position_x(3)
blob_position_x(3) = blob_position_x(4)
blob_position_x(4) = blob_position_x(5)
blob_position_x(5) = blob_position_x(6)
blob_position_x(6) = blob_position_x(7)
blob_position_x(7) = blob_position_x(8)
blob_position_x(8) = blob_position_x(9)
blob_position_x(9) = blob_position_x(10)
blob_position_x(10) = blob_position_x(11)
blob_position_x(11) = blob_position_x(12)
blob_position_x(12) = blob_position_x(13)
blob_position_x(13) = blob_position_x(14)
blob_position_x(14) = blob_position_x(15)
blob_position_x(15) = blob_position_x(16)
blob_position_x(16) = blob_position_x(17)
blob_position_x(17) = blob_position_x(18)
blob_position_x(18) = blob_position_x(19)
blob_position_x(19) = blob_position_x(20)
blob_position_y(1) = blob_position_y(2)
blob_position_y(2) = blob_position_y(3)
blob_position_y(3) = blob_position_y(4)
blob_position_y(4) = blob_position_y(5)
blob_position_y(5) = blob_position_y(6)
blob_position_y(6) = blob_position_y(7)
blob_position_y(7) = blob_position_y(8)
blob_position_y(8) = blob_position_y(9)
blob_position_y(9) = blob_position_y(10)
blob_position_y(10) = blob_position_y(11)
blob_position_y(11) = blob_position_y(12)
blob_position_y(12) = blob_position_y(13)
blob_position_y(13) = blob_position_y(14)
blob_position_y(14) = blob_position_y(15)
blob_position_y(15) = blob_position_y(16)
blob_position_y(16) = blob_position_y(17)
blob_position_y(17) = blob_position_y(18)
blob_position_y(18) = blob_position_y(19)
blob_position_y(19) = blob_position_y(20)
blob_lastseen(1) = blob_lastseen(2)
blob_lastseen(2) = blob_lastseen(3)
blob_lastseen(3) = blob_lastseen(4)
blob_lastseen(4) = blob_lastseen(5)
blob_lastseen(5) = blob_lastseen(6)
blob_lastseen(6) = blob_lastseen(7)
blob_lastseen(7) = blob_lastseen(8)
blob_lastseen(8) = blob_lastseen(9)
blob_lastseen(9) = blob_lastseen(10)
blob_lastseen(10) = blob_lastseen(11)
blob_lastseen(11) = blob_lastseen(12)
blob_lastseen(12) = blob_lastseen(13)
blob_lastseen(13) = blob_lastseen(14)
blob_lastseen(14) = blob_lastseen(15)
blob_lastseen(15) = blob_lastseen(16)
blob_lastseen(16) = blob_lastseen(17)
blob_lastseen(17) = blob_lastseen(18)
blob_lastseen(18) = blob_lastseen(19)
blob_lastseen(19) = blob_lastseen(20)
End sub
Public Sub comms()
do
debug.print "cat1:"&cstr(cat_x(1)) & " " & "cat2:"&cstr(cat_x(2)) & " " & "cat3:"&cstr(cat_x(3)) & " " & "cat4:"&cstr(cat_x(4))
call sleep(0.1)
loop
End Sub