Usage
controller
takes path
stream from imaging
service as input and produces a decision
output stream.
Input
path
stream has the following format:
pb_output.SensorOutput{
SensorId: 25,
Timestamp: uint64(time.Now().UnixMilli()),
SensorOutput: &pb_output.SensorOutput_CameraOutput{
CameraOutput: &pb_output.CameraSensorOutput{
DebugFrame: &pb_output.CameraSensorOutput_DebugFrame{
Jpeg: imgBytes.GetBytes(),
Canvas: &canvas,
},
Trajectory: &pb_output.CameraSensorOutput_Trajectory{
Points: trajectory_points,
Width: uint32(imgWidth),
Height: uint32(imgHeight),
},
},
},
}
SensorOutput_CameraOutput
object contains a CameraSensorOutput
object with 2 main fields:
-
CameraSensorOutput_DebugFrame
containing:Jpeg
, a (prerpocessed) acquired frame transmitted as a raw byte array of a JPEG image and thecanvas
, contains awidth
andheight
fields, as well as an array ofCanvasObject
s, which are either aLine
,Rectangle
orCircle
. It can be used for example to overlay these shapes on top of the JPEG image.
-
CameraSensorOutput_Trajecotry
is composed of:trajectory_points
array of points supplied by the imaging (default implementation only passes 1 point to the array).Width
of the acquired image.Height
of the acquired image
Processing input
Taken from src/main.go
:
imagingData := data.GetCameraOutput()
This unmarshals the SensorOutput
object and returns a CameraSensorOutput
object
trajectory := imagingData.GetTrajectory()
Next, we further process the input to extract a CameraSensorOutput_Trajectory
object to further work with.
how to use first point
The first point in the array of trajectory points is the identified middle of the visible part of the track. (see image below). You can, for example, compare value of its x
coordinate to the middle of the frame to see if the car is offset towards the left or right side of the track.
Controller
The default implementation of the controller
service uses a PID algorithm to make a decision on where to go (What is a PID controller?). It is very much the core of this service and is the primary part you would want to replace, should you decide to implement your own algorithm.
Output
controller
outputs the decision
stream (see example below), which is later read and interpreted by the actuator
.
actuatorOutput.Write(
&pb_outputs.SensorOutput{
SensorId: 2,
Timestamp: uint64(time.Now().UnixMilli()),
SensorOutput: &pb_outputs.SensorOutput_ControllerOutput{
ControllerOutput: &pb_outputs.ControllerOutput{
SteeringAngle: float32(steerValue),
LeftThrottle: float32(speed),
RightThrottle: float32(speed),
FrontLights: false,
},
},
},
)
SensorOutput_ControllerOutput
contains a ControllerOutput
object, composed of the following fields:
SteeringAngle
, afloat32
value between -1 (left) and 1 (right)LeftThrottle
, afloat32
value between -1 (full reverse) and 1 (full forward)RightThrottle
, afloat32
value between -1 (full reverse) and 1 (full forward)FrontLights
, aboolean
value previously used to turn on the front lights in the dark, currently not used
For seeing an example of using decision
stream, please look at actuator documentation