404
+ +Page not found
+ + +diff --git a/.nojekyll b/.nojekyll new file mode 100644 index 00000000..e69de29b diff --git a/404.html b/404.html new file mode 100644 index 00000000..a0655695 --- /dev/null +++ b/404.html @@ -0,0 +1,193 @@ + + +
+ + + + +Page not found
+ + +This generally means:
+Other options are perhaps possible, see the end of this document.
+Many car builders end up looking at "integrated" RC hobby cars, because they +are typically cheaper. However, the reason these are cheaper, is that they will +integrate many parts of electronics and mechanics into a single package, which +means that we can't intersect the appropriate signals to control the car with a +Raspberry Pi. In fact, the expected signals may not even exist at all in an +integrated car.
+Here is an example of an integrated RX and ESC - typically these should be avoided: +
+You also need to know some things about electronics, such as the difference +between power rails and control signals, what the duration of a microsecond is, +and how Volts, Amperes, Watts, Hours, Ohms, and other measurement units relate.
+While there are lots of designs out there besides the Donkeycar, but two stand out and are worth mentioning specifically.
+This is a flexible mounting system developed by Markku.ai.
+ +Doug LaRue, a long time community member has extensive designs for making your own chassis in sCAD. If you want to roll your own but are not comfortable with CAD this is a good place to start.
+ +An RC servo is used for controlling the steering wheels of the car. This servo +typically expects around 4.8V to 6V input on the power wire (varies by car) and +a PWM control signal on the signal wire. Typically, the three wires are colored +black-red-white, or brown-red-yellow, where the dark wire (black/brown) is ground, +and the center wire (red) is power, and the light wire (white/yellow) is control.
+The control signal is RC-style PWM, where one pulse is sent 60 times a second, +and the width of this pulse controls how left/right the servo turns. When this +pulse is 1500 microseconds, the servo is centered; when the pulse is 1000 +microseconds, the servo is turned all the way left (or right) and when the pulse +is 2000 microseconds, the servo is turned all the way in the other direction. +This is NOT the same kind of PWM that you would use to control the duty cycle of +a motor, or the brightness of a LED.
+The power for the servo typically comes from the motor ESC, which has a BEC +(Battery Eliminator Circuit) built in.
+The role of the ESC is to take a RC PWM control signal (pulse between 1000 and +2000 microseconds) in, and use that to control the power to the motor so the +motor spins with different amounts of power in forward or reverse. Again, 1500 +microseconds typically means "center" which for the motor means "dead stop."
+The battery typically connects straight to the ESC using thicker wiring than the +simple control signals, because the motor draws many more amps than the control. +The ESC then connects on to the motor with equally thick power wiring. The +standard Donkey motor and ESC probably have a peak current of about 12A; a +1/8th scale RC car with powerful brushless motor can have a peak draw up to +200A!
+Additionally, the ESC typically contains a linear or switching voltage converter +that outputs the power needed to control the steering servo; this is typically +somewhere in the 4.8V to 6V range. Most BECs built into ESCs will not deliver +more than about 1A of current, so it is not typically possible to power both the +steering servo and the Raspberry Pi from the BEC.
+If you buy a "kit car" that is listed as "needs a receiver," then you don't need +to buy a receiver. The Raspberry Pi plus the PCA9685 board take the role of the +receiver, outputting control signals to the car. Buying a "kit car" that comes +with steering servo, motor, and ESC, but not with radio, is actually a great way +to make sure that the car you build will have the right signalling, because any +RC car with separate receiver will be designed for the appropriate PWM signals.
+If your car comes with a receiver, make sure it has the appropriate three-pin +headers next to each other for steering servo and for ESC control. Some receivers +may have additional three-pin headers for additional channels, which may be empty +or may control fancy attachments like horns, lights, and so forth.
+There is a modification to the Donkey car which uses the RC radio to drive the +car when collecting training data; this will give better control of the car than +you typically get with a PlayStation controller, or cell phone. However, it also +requires replacing the PCA9685 board with an external micro-controller, and +changing the software of the Donkey to use it.
+Finally, some receivers can output, in addition to the PWM control signals, a +serial data packet that contains the control signals. An example of such a receiver +is the FS-i6B, which has 6 output channels for PWM signals, but can output 10 +channels of data at 115,200 bps as serial data, which you can read with an external +micro-controller, or perhaps even with the Raspberry Pi (requires re-configuration +of the Pi boot loader, and custom modifications to the donkey software.)
+The Donkey comes with a Nickel Metal Hydride battery (NiMH) which is just enough +to make its motor go, for a little bit of time (5-10 minutes) before needing a +recharge. The specifications on this battery are 6 cells, 1100 mAh. Because +NiHM batteries range from 0.9V to 1.35V with a "nominal" voltage of 1.2V, you can +expect to see voltages in the 5.4V to 8.1V range.
+NiHM batteries have medium energy capacity per weight and volume. Thus, you can +improve the runtime and performance of the Magnet car by upgrading to a Lithium +Polymer battery (LiPo.) Typically, you will get a 2 cell battery (2S) and +Lithium batteries have 3.2V to 4.2V per cell, so you will see voltages in the +6.4V to 8.4V range. Additionally, Lithium Polymer batteries generally have higher +current capacity (amount of Amps the battery can deliver at one point while +driving) as well as energy storage (number of Amp Hours the battery stores when +fully charged) so it may also last longer.
+Note that the amount of charge a battery can hold (how long it runs) is measured +in Ampere-hours (Ah), or milli-Ampere-hours (mAh), whereas the amount of current a battery +can instantaneously deliver while driving is measured simply in Amperes. But to +make things more confusing, Amperes are often re-calculated in terms of multiples +of the energy content, divided by one hour; this ratio is often called "C." Thus, +a LiPo rated for 10C and 2000 mAh, can deliver 20 Amperes of current while +driving. A NiHM rated for 5C and 1100 mAh can deliver 5.5 Amperes of current while +driving. Batteries typically will deliver more than the C rating for very short +amounts of time, but will heat up or build up internal resistance such that that +is not something you can rely on for normal operation.
+For your custom car, be aware of the voltages needed for the ESC and motor of the +car, and make sure to get a battery that matches in voltage. Smaller RC cars will +come with NiMH for affordability, or 2S LiPo for power. Larger RC cars will use 3S +(11.1V) or 4S (14.8V) or even 6S (22.2V) Lithium batteries, and thus need to have +ESC and motor combinations to match.
+Finally, be sure to get a charger that matches your battery. If you have a LiPo +battery, get a good Lithium battery charger, with a balancing plug that matches +your battery. Never discharge a Lithium battery below 3.2V per cell; if you let it +run dead, it will not want to be charged up to normal voltage again, and trying to +do so may very well overheat the batter and light it on fire! See YouTube pictures +of burning Teslas for what that can look like. Seriously, houses have burned down +because people have tried to save $10 by re-charging a Lithium battery that they +forgot to disconnect and it ran down too much. It's not worth it. Instead, get a +battery alarm, that you plug into the battery balance connector, and it beeps when +the battery has discharged so much that you should disconnect and recharge it.
+Adding the additional battery and electronics for self-driving to a toy car will +add more load than the car was initially designed for. For a large, 1/8th scale +car, this may not be much of a problem. For a small car, 1/18th scale or below, the +additional weight and top-heaviness will cause the car to not react well to the +steering output, which may cause the self-driving model to be less able to control +the car.
+If you use a car that's not the standard Magnet, at a minimum, you will have to +figure out how to mount all the hardware securely. Just piling things on and hoping +wiring will keep it in place will not work for things that actually drive and turn. +Finding good mounting points, and making your own "base plate" with measurements +from the car you have, is likely to be necessary. You can build this base plate +using 3D printing, laser cutting, CNC milling, or even just drilling into a thin +piece of plywood, but getting a good fit to your chassis is important, so don't +rush it or try to cut corners.
+Doug LaRue also built a configurator in Thingiverse that enables people to easily make custom 3D printed plates.
+Yes, you can make a self-driving car out of your 1/5th scale Nitro Dragster. You +will just have to learn even more about the different bits and pieces of the +solution, and figure out all the necessary integration yourself. The control +signals for a Nitro car are the same, so this might not even be hard. However, the +indoors arenas used for Donkey Racing Meetups do not allow fuel-burning cars, only +electric.
+Yes, you can make a self-driving car out of a cheap two-wheel chassis that uses +a LM298 H-bridge with direct PWM control to "tank steer" two wheels. However, you +will have to adapt the Donkey software to output the right steering controls, and +you will additionally have to figure out how to wire up the H-bridge to the Pi in +a way that makes sense to you; the PWM signals output by the PCA9685 board are the +RC control kind, NOT the motor control kind! Also, most affordable two-wheel-drive +robot chassis are not actually big enough, strong enough, and mechanically +consistent enough to make for good Donkey Car candidates.
+ +The magnet chassis was the first standard Donkey build. However in many cases it may not be available. +
+Try searching for both the Magnet and HSP 94186 on ebay, banggood, ali express etc.
+The HSP 94186 is the same as the Magnet and will work. If you speak mandarin it is always available on Taobao.
+ +The Desert Monster, SCT and Blaze are made by the same manufacturer as the Magnet and has the same motor and ESC. The chassis is slightly different so it requires an adapter and some extra hardware to work with the standard donkey platform. With the adapters the camera placement will be identical to the Magnet and should be able to share models.
+It is worth noting that the Desert Monster and SCT also has some nice characteristics including narrower, more road friendly tires and the Blaze has a slightly narrower stance which makes it less likely to hit things.
++To purchase one of these cars follow the following links:
+To assemble one of these you will need some additional parts than the standard build, these can be purchased as a kit on the donkey store at: Purchase: Donkey Store
+Part Description | +Link | +Approximate Cost | +
---|---|---|
3D printed Adapters | +Files: thingiverse.com/thing:2260575 | +$10 | +
Chassis Clips | +Amazon | +$5 | +
To assemble first remove the plastic cover and roll cage then unscrew the posts that hold up the cover and replace with the adapters.
+Visual instructions to follow.
+The LaTrax prerunner is a supported car and follows the same build instructions as the Desert Monster. However the adapters get screwed in as is shown in the photo below.
+ +To build a donkey pro the following parts are Needed
+Part Description | +Link | +Approximate Cost | +
---|---|---|
Donkey Pro Plastics and base | +Thingiverse or Donkeystore | +$50 | +
(8) M2.5 standoff | ++ | + |
(8) M2.5 Nylock nuts | ++ | + |
(8) M2.5x6mm socket head cap screws | ++ | + |
(4) M3x10 plastic self threading screw | ++ | + |
To assemble the Raspberry pi to the chassis this assembly picture should clarify how it fits together.
+ +The TT-01 is a new build that is a higher end version of the Donkey. This is an advanced build and requires existing RC skills or the desire to learn them - along with some willingness to trial and error. For first time builders we recommend the Magnet. That said, it has some pros and cons that people should be aware of, presented below.
+Pros:
+Cons:
+In addition to the standard donkey parts, Raspberry Pi etc, you will need to buy the following components.
+Part Description | +Link | +Approximate Cost | +
---|---|---|
TT-01 Clone Chassis | +amazon other TT01s may be used | +$130 | +
ESC | +Hobbyking | +10.60 | +
Brushed Motor | +Hobbyking | +$5 | +
Steering Servo | +Hobbyking | +$5 | +
Battery | +Hobbyking or similar 2S 5000 mAh battery | +$21 | +
Pinion Gear | +Amazon | +$7 | +
TT01 Plastics | +Thingiverse or Donkeystore | +$50 | +
Note: purchasing from Hobbyking is tricky. They can ship from multiple warehouses and it can be expensive and time consuming if shipping from one overseas. You may need to buy an alternate component if one of the items above are not available in your local warehouse.
+It's totally possible to diverge from the main Donkey build, and still have a car that +drives well and is fun to work with. We've seen a large variety of cars in the various +Donkey competitions around the world.
+However, when you want to diverge, there are several things you need to know, or you +will not be successful. There are many cost and quality trade-offs where the lower +cost options simply won't work. We've already worked hard to find the cheapest +available options that will work, so you should not expect to choose other options to +save money. Rolling your own is more about learning, experimentation, and going to new +and uncharged places.
+To find out more about what you need, see Roll Your Own.
+ +Donkey is an open source project to help accelerate the development of +self driving autos.
+There is a very good explanation of the DonkeyCar software architecture and theory here
+These guidelines are nearly copied from Keras, + because they are so good
+Are you a hardware specialist that can write a donkey part wrapper for a +GPS unit or a data scientist that can write an recursive neural net autopilot? +If so please write a part so other people driving donkeys can use the part. How do parts work? Check out this overview
+If you find a problem with the code and you know how to fix it then please +clone the repo, make your fix, and submit your pull request.
+Helping close or triage the issues is a good way to help.
+Search the code or docs for TODO
to find places where you might be able
+to find a better solution.
You can fix grammar or provide clarity by clicking the the Edit on GitHub +link in the top right corner. Here's a guide to how to create and edit docs.
+ +Thank you for contributing to the Donkeycar project. The documentation is critical for the success of our users so we appreciate your contributions. Accuracy and completeness is critical. Many users are beginners so please write your contributions with this in mind; don't assume that 'they should already know that'.
+We use the mkdocs package to create the html for the https://docs.donkeycar.com site. The files in the repo are in markdown format; mkdocs compiles
those to html so they can be displayed in a browser. You make your changes in your own fork of the donkeydocs repo and open a pull request so it can be merged into the main donkeydocs repo by one of the maintainers. Once the PR is merged then the changes will automatically be compiled and pushed to the https://docs.donkeycar.com site.
Make the changes/additions and check them in your fork. We use a package called mkdocs to compile the markdown files that you edit/create into html. See the mkdocs documentation for the particulars of the markdown format that it uses. If you install mkdocs you can use it to generate a live preview so you can see the changes as you save them.
+python3 -m venv env
+source env/bin/activate
+pip3 install mkdocs
+mkdocs serve
Once you are done making changes/addtions in your branch, commit the changes and push them to your forked repo. If you find that you need to make more changes then just rinse and repeat; make changes, commit them, push them.
+Compare & Pull Request
button; you can push that to create your pull request.This process is documented in more detail here https://docs.github.com/en/get-started/quickstart/contributing-to-projects
+ +Note: This requires version >= 4.1.X
+You might want to write your own model:
+Models are located in donkeycar/parts/keras.py
. Your own model needs to
+inherit from KerasPilot
and initialize your model:
class KerasSensors(KerasPilot):
+ def __init__(self, input_shape=(120, 160, 3), num_sensors=2):
+ super().__init__()
+ self.num_sensors = num_sensors
+ self.model = self.create_model(input_shape)
+
+Here, you implement the keras model
+in the member function create_model()
. The model needs to have labelled input
+and output tensors. These are required for the training to work.
What is required for your model to work, are the following functions:
+def compile(self):
+ self.model.compile(optimizer=self.optimizer, metrics=['accuracy'],
+ loss={'angle_out': 'categorical_crossentropy',
+ 'throttle_out': 'categorical_crossentropy'},
+ loss_weights={'angle_out': 0.5, 'throttle_out': 0.5})
+
+The compile
function tells keras how to define the loss function for training.
+We are using the KerasCategorical
model as an example. The loss function here
+makes explicit usage of the output tensors of the
+model (angle_out, throttle_out
).
def x_transform(self, record: TubRecord):
+ img_arr = record.image(cached=True)
+ return img_arr
+
+In this function you define how to extract the input data from your
+recorded data. This data is usually called X
in the ML frame work . We have
+shown the implementation in the base class which works for all models that have
+only the image as input.
The function returns a single data item if the model has only one input. You +need to return a tuple if your model uses more input data.
+Note: If your model has more inputs, the tuple needs to have the image in +the first place.
+def y_transform(self, record: TubRecord):
+ angle: float = record.underlying['user/angle']
+ throttle: float = record.underlying['user/throttle']
+ return angle, throttle
+
+In this function you specify how to extract the y
values (i.e. target
+values) from your recorded data.
def x_translate(self, x: XY) -> Dict[str, Union[float, np.ndarray]]:
+ return {'img_in': x}
+
+Here we require a translation of how the X
value that you extracted above will
+be fed into tf.data
. Note, tf.data
expects a dictionary if the model has
+more than one input variable, so we have chosen to use dictionaries also in the
+one-argument case for consistency. Above we have shown the implementation in the
+base class which works for all models that have only the image as input. You
+don't have to overwrite neither x_transform
nor x_translate
if your
+model only uses the image as input data.
Note: the keys of the dictionary must match the name of the input +layers in the model.
+def y_translate(self, y: XY) -> Dict[str, Union[float, np.ndarray]]:
+ if isinstance(y, tuple):
+ angle, throttle = y
+ return {'angle_out': angle, 'throttle_out': throttle}
+ else:
+ raise TypeError('Expected tuple')
+
+Similar to the above, this provides the translation of the y
data into the
+dictionary required for tf.data
. This example shows the implementation of
+KerasLinear
.
Note: the keys of the dictionary must match the name of the output +layers in the model.
+def output_shapes(self):
+ # need to cut off None from [None, 120, 160, 3] tensor shape
+ img_shape = self.get_input_shape()[1:]
+ shapes = ({'img_in': tf.TensorShape(img_shape)},
+ {'angle_out': tf.TensorShape([15]),
+ 'throttle_out': tf.TensorShape([20])})
+ return shapes
+
+This function returns a tuple of two dictionaries that tells tensorflow which
+shapes are used in the model. We have shown the example of the
+KerasCategorical
model here.
Note 1: As above, the keys of the two dictionaries must match the name +of the input and output layers in the model.
+Note 2: Where the model returns scalar numbers, the corresponding
+type has to be tf.TensorShape([])
.
In the car application the model is called through the run()
function. That
+function is already provided in the base class where the normalisation of the
+input image is happening centrally. Instead, the derived classes have to
+implement
+inference()
which works on the normalised data. If you have additional data
+that needs to be normalised, too, you might want to override run()
as well.
def inference(self, img_arr, other_arr):
+ img_arr = img_arr.reshape((1,) + img_arr.shape)
+ outputs = self.model.predict(img_arr)
+ steering = outputs[0]
+ throttle = outputs[1]
+ return steering[0][0], throttle[0][0]
+
+Here we are showing the implementation of the linear model. Please note that
+the input tensor shape always contains the batch dimension in the first
+place, hence the shape of the input image is adjusted from
+(120, 160, 3) -> (1, 120, 160, 3)
.
Note: _If you are passing another array in theother_arr
variable, you will
+have to do a similar re-shaping.
Let's build a new donkey model which is based on the standard linear model +but has following changes w.r.t. input data and network design:
+The model takes an additional vector of input data that represents a set + of values from distance sensors which are attached to the front of the car.
+The model adds a couple of more feed-forward layers to combine the CNN + layers of the vision system with the distance sensor data.
+So here is the example model:
+class KerasSensors(KerasPilot):
+ def __init__(self, input_shape=(120, 160, 3), num_sensors=2):
+ super().__init__()
+ self.num_sensors = num_sensors
+ self.model = self.create_model(input_shape)
+
+ def create_model(self, input_shape):
+ drop = 0.2
+ img_in = Input(shape=input_shape, name='img_in')
+ x = core_cnn_layers(img_in, drop)
+ x = Dense(100, activation='relu', name='dense_1')(x)
+ x = Dropout(drop)(x)
+ x = Dense(50, activation='relu', name='dense_2')(x)
+ x = Dropout(drop)(x)
+ # up to here, this is the standard linear model, now we add the
+ # sensor data to it
+ sensor_in = Input(shape=(self.num_sensors, ), name='sensor_in')
+ y = sensor_in
+ z = concatenate([x, y])
+ # here we add two more dense layers
+ z = Dense(50, activation='relu', name='dense_3')(z)
+ z = Dropout(drop)(z)
+ z = Dense(50, activation='relu', name='dense_4')(z)
+ z = Dropout(drop)(z)
+ # two outputs for angle and throttle
+ outputs = [
+ Dense(1, activation='linear', name='n_outputs' + str(i))(z)
+ for i in range(2)]
+
+ # the model needs to specify the additional input here
+ model = Model(inputs=[img_in, sensor_in], outputs=outputs)
+ return model
+
+ def compile(self):
+ self.model.compile(optimizer=self.optimizer, loss='mse')
+
+ def inference(self, img_arr, other_arr):
+ img_arr = img_arr.reshape((1,) + img_arr.shape)
+ sens_arr = other_arr.reshape((1,) + other_arr.shape)
+ outputs = self.model.predict([img_arr, sens_arr])
+ steering = outputs[0]
+ throttle = outputs[1]
+ return steering[0][0], throttle[0][0]
+
+ def x_transform(self, record: TubRecord) -> XY:
+ img_arr = super().x_transform(record)
+ # for simplicity we assume the sensor data here is normalised
+ sensor_arr = np.array(record.underlying['sensor'])
+ # we need to return the image data first
+ return img_arr, sensor_arr
+
+ def x_translate(self, x: XY) -> Dict[str, Union[float, np.ndarray]]:
+ assert isinstance(x, tuple), 'Requires tuple as input'
+ # the keys are the names of the input layers of the model
+ return {'img_in': x[0], 'sensor_in': x[1]}
+
+ def y_transform(self, record: TubRecord):
+ angle: float = record.underlying['user/angle']
+ throttle: float = record.underlying['user/throttle']
+ return angle, throttle
+
+ def y_translate(self, y: XY) -> Dict[str, Union[float, np.ndarray]]:
+ if isinstance(y, tuple):
+ angle, throttle = y
+ # the keys are the names of the output layers of the model
+ return {'n_outputs0': angle, 'n_outputs1': throttle}
+ else:
+ raise TypeError('Expected tuple')
+
+ def output_shapes(self):
+ # need to cut off None from [None, 120, 160, 3] tensor shape
+ img_shape = self.get_input_shape()[1:]
+ # the keys need to match the models input/output layers
+ shapes = ({'img_in': tf.TensorShape(img_shape),
+ 'sensor_in': tf.TensorShape([self.num_sensors])},
+ {'n_outputs0': tf.TensorShape([]),
+ 'n_outputs1': tf.TensorShape([])})
+ return shapes
+
+We could have inherited from KerasLinear
which already provides the
+implementation of y_transform(), y_translate(), compile()
. However, to
+make it explicit for the general case we have implemented all functions here.
+The model requires the sensor data to be an array in the TubRecord with key
+"sensor"
.
Because we don't have a tub with sensor data, let's create one with fake +sensor entries:
+import os
+import tarfile
+import numpy as np
+from donkeycar.parts.tub_v2 import Tub
+from donkeycar.pipeline.types import TubRecord
+from donkeycar.config import load_config
+
+
+if __name__ == '__main__':
+ # put your path to your car app
+ my_car = os.path.expanduser('~/mycar')
+ cfg = load_config(os.path.join(my_car, 'config.py'))
+ # put your path to donkey project
+ tar = tarfile.open(os.path.expanduser(
+ '~/Python/donkeycar/donkeycar/tests/tub/tub.tar.gz'))
+ tub_parent = os.path.join(my_car, 'data2/')
+ tar.extractall(tub_parent)
+ tub_path = os.path.join(tub_parent, 'tub')
+ tub1 = Tub(tub_path)
+ tub2 = Tub(os.path.join(my_car, 'data2/tub_sensor'),
+ inputs=['cam/image_array', 'user/angle', 'user/throttle',
+ 'sensor'],
+ types=['image_array', 'float', 'float', 'list'])
+
+ for record in tub1:
+ t_record = TubRecord(config=cfg,
+ base_path=tub1.base_path,
+ underlying=record)
+ img_arr = t_record.image(cached=False)
+ record['sensor'] = list(np.random.uniform(size=2))
+ record['cam/image_array'] = img_arr
+ tub2.write_record(record)
+
+We don't have a dynamic factory yet, so we need to add the new model into the
+function get_model_by_type()
in the module donkeycar/utils.py
:
...
+elif model_type == 'sensor':
+ kl = KerasSensors(input_shape=input_shape)
+...
+
+In your car app folder now the following should work:
+donkey train --tub data2/tub_sensor --model models/pilot.h5 --type sensor
+Because of the random values in the data the model will not converge quickly,
+the goal here is to get it working in the framework.
Please join the Discord Donkey Car group for +support and discussions.
+ +There is a limited test suite to ensure that the your changes to the code +don't break something unintended.
+Run pytest
from the donkeycar
project directory.
The test code is in tests
foders in the same folder as the code. This is to
+help keep the test code linked to the code its self. If you change the code,
+change the tests. :)
+
The latest version of the software installation instructions are maintained in the software instructions section. Be sure to follow those instructions after you've built your car.
+There are two main options for cars. One is the WL Toys brand and the second is the exceed brand.
+NOTE: only the WL Toys 144010 and HSP-94186 are readily avilable rignt now
+The WL Toys 144010 is probably the easiest car to get right now. It has a brushless motor, which makes it fast, but takes some getting used to for beginners. There are brushed motor versions of theis car the 144011 and 144001 but both require the user to replace both the steering servo and ESC. Only do this if you are familiar with RC or enjoy tinkering. Here is a short video explaining how to assemble the car. You can find the adapters in Thingiverse or if you would like to buy them you can do so one the donkey car store
+The alterate car, which often has slighly less availability is the HSP 94186 and the "Exceed" brand cars. There are 5 supported cars, all are very similar and should be considered equivalent. Note, often some of these are out of stock, so go through the links to find one that is in stock. If they are out of stock on Amazon, you can find the cars at the Exceed Website. The HSP-94186 is identical to the Exceed Magnet 1/16 Truck; it can be found on AliExpress but takes about a month to get to the US, there are local options that charge a premium.
+These cars are electrically identical but have different tires, mounting and other details. It is worth noting that the Desert Monster, Short Course Truck and Blaze all require adapters which can be easily printed or purchased from the donkey store. These are the standard build cars because they are mostly plug and play, both have a brushed motor which makes training easier, they handle rough driving surfaces well and are inexpensive.
+Here is a video overview of the different cars (Excluding the WL Toys car) and how to assemble them.
+For advanced users there are 2 more cars supported under the "Donkey Pro" name. These are 1/10 scale cars which means that they are bigger, perform a little better and are slightly more expensive. They can be found here:
+Here is a video that goes over the different models. The Donkey Pro models are not yet very well documented, just a word of warning.
+For more detail and other options, follow the link to: supported cars
+ +Alternatively If you know RC or need something the standard Donkey does not support, you can roll your own. Here is a quick reference to help you along the way. Roll Your Own
+This video covers how to assemble a standard Donkey Car, it also covers the Sombrero, the Raspberry Pi and the nVidia Jetson Nano.
+ +The following instructions are for the Raspberry Pi, below in Optional Upgrades section, you can find the NVIDIA Jetson Nano instructions.
+There are two official stores:
+If you are in the US, you can use the Donkey store. The intention of the Donkey Store is to make it easier and less expensive to build the Donkey Car. The Donkey Store is run by the original founders of donkey car and profits are used to fund development of the donkey cars. Also it is worth noting the design of the parts out of the Donkey store is slightly improved over the standard build as it uses better parts that are only available in large quantities or are harder to get. The Donkey Store builds are open source like all others.
+If you are in Asia, the DIYRobocars community in Hong Kong also sells car kits at Robocar Store. They are long term Donkey community members and use proceeds to support the R&D efforts of this project. It is worth noting they can also sell to Europe and the US but it is likely less cost effective.
+Part Description | +Link | +Approximate Cost | +
---|---|---|
WL Toys 144010, Exceed Magnet, Desert Monster, Blaze, or Short Course Truck | +See links above | +$100-130 | +
USB Battery with microUSB cable (any battery capable of 2A 5V output is sufficient) | +Anker 10,000 mAh | +$39 | +
Raspberry Pi 3b+ | +Pi 3b+ | +$42 | +
MicroSD Card (many will work, we strongly recommend this one) | +64GB https://amzn.to/2XP7UAa | +$11.99 | +
Donkey Partial Kit | +KIT | +$82 to $125 | +
If you want to buy the parts yourself, want to customize your donkey or live outside of the US, you may want to choose the bottoms up build. Keep in mind you will have to print the donkey car parts which can be found here
+Part Description | +Link | +Approximate Cost | +
---|---|---|
Magnet Car or alternative | +see cars above under 'choosing a car' | +$92 | +
M2x6 screws (8) | +Amazon or Donkey Store | +$4.89 * | +
M3x10 screws (3) | +Amazon or Donkey Store | +$7.89 * | +
USB Battery with microUSB cable (any battery capable of 2A 5V output is sufficient) | +Anker 10,000 maH | +$39 | +
Raspberry Pi 3b+ | +Pi 3B+ | +$38 | +
MicroSD Card (many will work, I like this one because it boots quickly) | +64GB | +$18.99 | +
Wide Angle Raspberry Pi Camera | +Amazon or Donkey Store | +$25 | +
Female to Female Jumper Wire | +Amazon or Donkey Car Store | +$7 * | +
(Optional if you don't want to use RPi GPIO pins to control the car's servo and throttle directly) Servo Driver PCA 9685 | +Amazon or Donkey Car Store | +$12 ** | +
3D Printed roll cage and top plate. | +Purchase: Donkey Store Files: thingiverse.com/thing:2260575 | +$50 | +
* If it is hard to find these components, there is some wiggle room. Instead of an M2 you can use an M2.2, m2.3 or #4 SAE screw. Instead of an M3 a #6 SAE screw can be used. Machine screws can be used in a pinch.
+** This component can be purchased from Ali Express for ~$2-4 if you can wait the 30-60 days for shipping.
+Plug in the Servo driver the same as the Raspberry Pi, just keep in mind that the Jetson pinout is reversed and that the Sombrero is not supported.
+ +Finally this is the Donkey Assembled.
+ +Part Description | +Link | +Approximate Cost | +
---|---|---|
Nvidia Jetson Nano | +Amazon | +$99 | +
Jetson Nano Adapter | +Donkey Store | +$7 | +
Camera Module | +Donkey Store | +$27 | +
WiFi Card | +Amazon | +$18 | +
Antennas | +Donkey Store | +$7 | +
For other options for part, feel free to look at the jetbot documentation here.
+Part Description | +Link | +Approximate Cost | +
---|---|---|
LiPo Battery | +hobbyking.com/en_us/turnigy-1800mah-2s-20c-lipo-pack.html or amazon.com/gp/product/B0072AERBE/ | +$8.94 to $~17 | +
Lipo Charger (takes 1hr to charge the above battery) | +charger | +$13 | +
Lipo Battery Case (to prevent damage if they explode) | +lipo safe | +$8 | +
If you purchased parts from the Donkey Car Store, skip to step 3.
+If you do not have a 3D Printer, you can order parts from Donkey Store, Shapeways or 3dHubs. I printed parts in black PLA, with 2mm layer height and no supports. The top roll bar is designed to be printed upside down. Remember that you need to print the adapters unless you have a "Magnet"
+I printed parts in black PLA, with .3mm layer height with a .5mm nozzle and no supports. The top roll bar is designed to be printed upside down.
+Almost all 3D Printed parts will need clean up. Re-drill holes, and clean up excess plastic.
+ +In particular, clean up the slots in the side of the roll bar, as shown in the picture below:
+ +If you have an Exceed Short Course Truck, Blaze or Desert Monster watch this video
+This is a relatively simple assembly step. Just use the 3mm self tapping screws to scew the plate to the roll cage.
+When attaching the roll cage to the top plate, ensure that the nubs on the top plate face the roll-cage. This will ensure the equipment you mount to the top plate fits easily.
+note: this is not necessary if you are using direct control with RaspberyPi GPIO pins or are using a Robohat MM1 board
+The PCA9685 Servo controller can control up to 16 PWM devices like servos, motor controllers, LEDs or almost anything that uses a PWM signal. It is connected to the RaspberryPi (or Jetson Nano) 40 pin GPIO bus via the I2C pins.
+---
+ GPIO ... PCA9685 ... 5v ... ESC ... Servo
+ 3v3-01 <---> VCC
+ pin-03 <---> SDA
+ pin-05 <---> SCL
+ GND-09 <---> GND
+ VIN <---> 5v optional, see above
+ GND <---> GND
+ CH-0 <---------> ESC
+ CH-1 <------------------> Servo
+---
+
+---
+ $ i2cdetect -y -r 1
+ 0 1 2 3 4 5 6 7 8 9 a b c d e f
+ 00: -- -- -- -- -- -- -- -- -- -- -- -- --
+ 10: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+ 20: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+ 30: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+ 40: 40 -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+ 50: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+ 60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+ 70: UU -- -- -- -- -- -- --
+---
+
+You could do this after attaching the Raspberry Pi to the bottom plate, I just think it is easier to see the parts when they are laying on the workbench. Connect the parts as you see below:
+ +For reference, below is the Raspberry Pi Pinout for reference. You will notice we connect to 3.3v, the two I2C pins (SDA and SCL) and ground:
+ +Before you start, now is a good time to insert the already flashed SD card and bench test the electronics. Once that is done, attaching the Raspberry Pi and Servo is as simple as running screws through the board into the screw bosses on the top plate. The M2.5x12mm screws should be the perfect length to go through the board, the plastic and still have room for a washer. The “cap” part of the screw should be facing up and the nut should be on the bottom of the top plate. The ethernet and USB ports should face forward. This is important as it gives you access to the SD card and makes the camera ribbon cable line up properly.
+Attach the USB battery to the underside of the printed bottom plate using cable ties or velcro.
+ +Slip the camera into the slot, cable end first. However, be careful not to push on the camera lens and instead press the board. +
+If you need to remove the camera the temptation is to push on the lens, instead push on the connector as is shown in these pictures.
+
Before using the car, remove the plastic film or lens cover from the camera lens.
+ +It is easy to put the camera cable in the wrong way so look at these photos and make sure the cable is put in properly. There are loads of tutorials on youtube if you are not used to this.
+ +*** Note if you have a Desert Monster Chassis see 7B section below *** +The final steps are straightforward. First attach the roll bar assembly to the car. This is done using the same pins that came with the vehicle.
+ +Second run the servo cables up to the car. The throttle cable runs to channel 0 on the servo controller and steering is channel 1.
+ +Now you are done with the hardware!!
+The Desert monster does not have the same set up for holding the body on the car and needs two adapters mentioned above. To attach the adapters you must first remove the existing adapter from the chassis and screw on the custom adapter with the same screws as is shown in this photo:
+ +Once this is done, go back to step 7
+Congrats! Now to get your get your car moving, see the software instructions section.
+ +++ +We are a participant in the Amazon Services LLC Associates Program, an affiliate advertising program designed to provide a means for us to earn fees by linking to Amazon.com and affiliated sites.
+
The point of calibrating your car is to make it drive consistently. If you have a steering servo then donkey needs to know the PWM values associated with full left and full right turns. If you have an ESC, then donkey needs to know the PWM values for full forward throttle, stopped and full reverse throttle. You figure out those values in the calibration process, then save them to your myconfig.py file so they can be used then the car is driving.
+Some kinds of drivetrains do not need to be calibrated. If you are using any drivetrain that uses an L298N motor controller or similar (rather than and ESC), then no calibration is necessary; those drivetrains do not use PWM; they use a duty cycle that does not need to be calibrated. Most of the differential drivetrains (those whose name begins with DC_TWO_WHEEL
) are of that type. If your drivetrain uses an L298N motor controller or similar for throttle, but uses a servo for steering, then you will need to calibrate steering, but not throttle.
There is a more complete discussion of drivetrains in Actuators
+++You will need to ssh into your Pi to do the calibration.
+
All of the car's default settings are in the config.py
. You can override the default settings by editing the myconfig.py
script in your car directory. This was generated when you ran the donkey createcar --path ~/mycar
command. You can edit this file on your car by running:
nano ~/mycar/myconfig.py
+
+++Make sure your car is off the ground to prevent a runaway situation.
+
donkey calibrate ...
and provide it with arguments to specify which pin will produce the PWMPWM_STEERING_THROTTLE
, then use --pwm-pin
argument to specify the target pin, like RPI_GPIO.BOARD.33
or PCA9685.1:40.13
. If you are using the Donkey Hat then you would use donkey calibrate --pwm-pin=PIGPIO.BCM.13
to calibrate steering. See Pins for a more complete discussion of pins and pin specifiers.I2C_SERVO
, then specify the PCA9685 channel (the index of the 3-pin connector that cable in connected to) and the I2C bus the PCA9685 is connected to; donkey calibrate --channel <your_steering_channel> --bus=<your_i2c_bus>
360
and you should see the wheels on your car move slightly. If not try 400
or 300
. Next enter values +/- 10 from your starting value to find the PWM setting that makes your car turn all the way left, again making sure the motor is not making a whining sound. Remember this value.Edit the myconfig.py
script on your car and enter these values as STEERING_LEFT_PWM
and STEERING_RIGHT_PWM
respectively.
STEERING_LEFT_PWM
= PWM for full left turnSTEERING_RIGHT_PWM
= PWM value for full right turndonkey calibrate ...
and provide it with arguments to specify which pin will produce the PWMPWM_STEERING_THROTTLE
, then use --pwm-pin
argument to specify the target pin, like RPI_GPIO.BOARD.33
or PCA9685.1:40.13
. If you are using the Donkey Hat then you would use donkey calibrate --pwm-pin=PIGPIO.BCM.18
to calibrate throttle. See Pins for a more complete discussion of pins and pin specifiers.I2C_SERVO
, then specify the PCA9685 channel (the index of the 3-pin connector that cable in connected to) and the I2C bus the PCA9685 is connected to; donkey calibrate --channel <your_throttle_channel> --bus=<your_i2c_bus>
370
when prompted for a PWM value.400
and you should see your cars wheels start to go forward. If not,
+its likely that this is reverse, try entering 330
instead.Reverse on RC cars is a little tricky because the ESC must receive a reverse pulse, zero pulse, reverse pulse to start to go backwards. To calibrate a reverse PWM setting...
+Now open your myconfig.py
script and enter the PWM values for your car into the throttle_controller part:
THROTTLE_FORWARD_PWM
= PWM value for full throttle forwardTHROTTLE_STOPPED_PWM
= PWM value for zero throttleTHROTTLE_REVERSE_PWM
= PWM value at full reverse throttleNow that you have your car roughly calibrated you can try driving it to verify that it drives as expected. Here's how to fine tune your car's calibration.
+First and most importantly, make sure your car goes perfectly straight when no steering input is applied.
+python manage.py drive
.<your_cars_hostname.local>:8887
in a browser.i
key on your keyboard a couple of times to get the car to move forward. This is best done if you have your car on a very flat floor with some kind of grid, so you can guage if it going straight. Be careful not to confuse driving off at an angle versus driving along an arc. Driving at an angle may simply mean you pointed the car at an angle when starting it. Driving a curved arc indicates the car is steering.STEERING_LEFT_PWM
in your myconfig.py file so it is closer to neutral. For example, if STEERING_LEFT_PWM
is 460 and STEERING_RIGHT_PWM
is 290, then reduce STEERING_LEFT_PWM
a little, maybe 458.STEERING_RIGHT_PWM
in your myconfig.py file so it is closer to neutral. For example, if STEERING_LEFT_PWM
is 460 and STEERING_RIGHT_PWM
is 290, then increase STEERING_RIGHT_PWM
a little, maybe 292.Next, try to make it so that a full left turn and a full right turn are the same turn angle (they make the same diameter circle when driven all the way around).
+++Note : optional
+
python manage.py drive
.<your_cars_hostname.local>:8887
in a browser.j
until the cars steering is all the way right.i
a couple of times to get the car to go forward.Corrections:
+After you've fine tuned your car the steering chart should look something like this.
+ +You may need to iterate making sure the car is driving straight and that the left and right turns are the same to get those both to work. Prioritize making sure the car drives straight.
+The computer vision autopilot, like the deep learning autopilot, interprets camera images in order to determine steering and throttle values. However, rather than deep learning models, the computer vision autopilot utilizes traditional computer vision algorithms, such as Canny edge detection, to interpret images of the track. The computer vision autopilot is specifically designed to make it easy to write your own algorithm and use it in place of the built-in algorithm.
+ +The built-in algorithm is a line following algorithm; it expects the track to have a center line, preferably solid, that it can detect. The expected color of the line can be tuned with configuration; by default it expects a yellow line. The algorithm calculates the distance of the line from the center of the image, then a PID controller uses that value to calculate a steering value. If the car is to the left of the line then it will turn right. If the car is to the right of the line then it will turn left. The chosen steering angle is proportional to the distance from the line. The chosen throttle is inversely proportional to the steering angle so that the car will go faster on a straight path and slow down for turns. More details on the algorithm and the configuration parameters are discussed below.
+But what if your track does not have a center line; what if it just has a left and right lane boundary lines? What if it is a sidewalk? What if you simply want to implement your own algorithm? The computer vision template is designed to make that pretty easy. You can write your own part in Python to use as the autopilot and simply change the configuration in your myconfig.py to point to it. Your part can utilize computer vision parts in cv.py or you can call OpenCV's python api directly. We present a simplified example below.
+++++IMPORTANT: The computer vision template requires that opencv is installed. Opencv is pre-installed on the Jetson Nano, but it must be explicitly installed on the Raspberry Pi. See Raspberry Pi installation Step 9 and Step 11.
+
You can create a computer vision application similarly to the how we create a deep learning application; we just tell it to use the cv_control template instead of the default template. First, make sure your donkeycar python environment is activated, then use the createcar command to create your application folder.
+donkey createcar --template=cv_control --path=~/mycar
+
+When updating to a new version of donkeycar, you will want to refresh your application folder. You can do this with the same command, but add --overwrite
so that it does not erase your myconfig.py file.
donkey createcar --template=cv_control --path=~/mycar --overwrite
+
+The built-in algorithm can follow a line using the camera. By default it is tuned for a yellow line, but the color that it tracks can be configured. Many other aspects of the algorithm can be tuned. Below is as description of the algorithm and how it uses the configuraton values. The values themselves are listed and described afterwards.
+TARGET_PIXEL
is None, then use steps 1 to 5 to estimate the target (the expected) position of the line.SCAN_Y
and SCAN_HEIGHT
pixels height. So the result is a block of pixels as wide as the image and SCAN_HEIGHT
high.RBG
red-gree-blue color space to HSV
hue-saturation-value color space. COLOR_THRESHOLD_LOW
and COLOR_THRESHOLD_HIGH
.SCAN_HEIGHT
pixel high slice.TARGET_PIXEL
value is used as the value the PID algorithm uses to calculate a new steering. If the value is to the left of the TARGET_PIXEL
more than TARGET_THRESHOLD
pixels then the car steers right; if hte value is to the right of TARGET_PIXEL
more than TARGET_THRESHOLD
pixels the the car steers left. If the value is withing TARGET_THRESHOLD
pixels of TARGET_PIXEL
then steering is not changed.THROTTLE_STEP
, but not over THROTTLE_MAX
. If steering is changed then throttle is decreased by THROTTLE_STEP
, but not below THROTTLE_MIN
.++++This pyimagesearch article and accompanying video describe the various color spaces available and OpenCV and their characteristics.
+
The complete source code is provided and discussed in the LineFollower class section near the end of this page.
+The image at the top of the page shows the camera setup approximately how it would be using the standard Donkeycar cage. It is angled to see to the horizon so that it can see turns from far away. This is good when going very fast because you can see far ahead. However if the detected line is very thin then it could have artifacting (noise) that could lead to false positives that cause the vehicle to move off the line. If you are not going fast and you want to be as accurate as possible then pointing the camera down at the line is a good idea. So if your camera can be adjusted then you can make trade-offs between accuracy (point it down) and speed (point to to the horizon).
+The computer vision template is a little different than the deep learning and path follow templates; there is no data recording. After setting your configuration parameters you just put your car on the track that has the line that you want to follow and then change from user mode to one of the auto-pilot modes; full-auto or auto-steering. The complete set of configuration parameters can be found in the LineFollower Configuration section below; we will discuss the most important configuration in more detail in this section.
+The rectangular area that will be scanned for the line, called the detection area, is determined with the SCAN_Y
and SCAN_HEIGHT
.
When in autopilot mode, the LineFollower
shows the detection area as a horizontal black bar. Pixels that fall within the color threshold range (see next sectino) are drawn as white pixels. Ideally, only the pixels in the line that are in the detection bar will show as white; any white pixels that are NOT part of the line that you want to follow are considered false positives. If the false positives are relatively disperse then they should not interfere with detecting the line. However, if there are big areas of white false positives then they might trick the algorithm. See the next section of how to adjust the color threshold range to minimize false positives.
The image below shows the detection area and the detected line.
+ +The color threshold values represent the range of colors used to detect the line; they should be chosen to include the colors in the line in the area that it passes through the detection bar and ideally they should not include any other colors. The color threshold values are in HSV color space (Hue, Saturation, Color) format, not RGB format. RGB color space is how a computer shows colors. HSV color space is closer to how humans perceive color. For our purposes the 'hue' part is the 'pure' color without regard for shadows or lighting. This makes it easier to find a color because it is one number, rather than combination of 3 numbers.
+++++There are many online converters between RGB and HSV. This one was used when creating this documentation; peko-step I like that tool because it will allow the Saturation and Value to be output in the range of 0.255, which is what we need. IMPORTANT: The online tools use the standard way of representing HSV, which is a Hue value of 0 to 359 degrees, Saturation of 0 to 100%, Value of 0 to 100%. OpenCV, which our code is based on, uses a Hue value of 0 to 179, Saturation of 0 to 255 and value of 0 to 255; so be aware that you may need to convert from the tool's values to the OpenCV values when changing these configurations.
+
When choosing the threshold colors it is important to take into account what the camera will see including the lighting conditions. Donkeycar includes a script to make this easy to do. the hsv_picker.sh
script allows you to view the live camera image or alternatively to choose a static image to view. So if you are running a desktop image on your car (so not a server image or headless image) then you can run the script and view the camera image. If you do not have a desktop on the car, then you can run the car and open the web view in a browser on your host laptop at take a screen shot to save it, then use that static image with the hsv_picker.sh
script on your laptop computer. In either case, arrange the car on the course so it can see the line as it would see it when you drive in autopilot so you are getting a realistic view.
You can run the hsv_picker.sh
script to view a screen shot image; with the donkey python environment activated run the script from the root of your donkeycar repo folder;
python scripts/hsv_picker.sh --file=<path-to-image>
+
+To view the camera stream, again with donkey python environment activated, run the script from the root of your donkeycar repo folder;
+python scripts/hsv_picker.sh
+
+If you have more than one camera and it is not showing the correct one, you can choose the camera index and/or set the image size
+python scripts/hsv_picker.sh --camera=2 --width=320 --height=240
+
+
+The image above shows the hsv_script.sh
with a web ui screenshot loaded. The blue line in the center of the image is the line that we want to follow. The horizontal black bar in the camera image is the detection bar; this is defined by SCAN_Y
and SCAN_HEIGHT
and is the area where the mask is applied to try to isolate the pixels in the line. When pixels are detected they will be draw in white in the detection area.
The bottom of the screen has 6 trackbars to select the 3 parts of the low HSV value and the 3 parts of the high HSV value that are used to create a mask to pull out the pixels in the line. You can move those scrollbars manually to try to find the best values for the detection range. As you change the scrollbars the resulting mask will be applied to the image and you will see pixels start to turn back. The Hue value is typically the most important value. You can reset the trackbars and clear the mask anytime by selecting the Escape key on the keyboard.
+Using the scrollbars works, but there is an easier way. You can also just select a rectangular area by clicking-dragging-releasing; the pixels in that area will be searched for a low and high value and the trackbars will by updated with those low and high values. So the easiest way to find the mask for the line is to select a rectangular area on the line itself. You can fine-tune the selected mask using the trackbars.
+The image below shows the mask that was created by selecting a rectangular area within the blue line.
+ +Features of hsv_picker.sh
:
The TARGET_PIXEL
value is the expected horizontal position of the line to follow in the image. The line follow algorithm will adjust steering to try to keep the line at that position in the image. More specifically, the difference between the TARGET_PIXEL
value and where the line follow algorithm detects that actual line is in the image is used by a PID controller to adjust steering (see The PID Controller) below.
If you are the only car on the course, then you probably want the car to drive directly on the line to follow. In this case setting TARGET_PIXEL
to the horizontal center of the image at (IMAGE_W / 2)
means the auto-pilot assumes the line to follow should be directly in the middle of the image and so the car will try to stay in the middle. So if your car actually starts to the left or right of that line, it will quickly move the the line and stay on it.
However, if you are on a course where two cars drive at the same time (there are two lanes separated by a line), then you probably want your car to stay in it's lane. In that case you would set TARGET_PIXEL
to None, which will cause the car to detect the location of the line at startup. The auto-pilot will then assume the line should stay at that position in the image, and so it will the try to keep the car it it's lane to make that true.
++++If you are really motivated then you might try implementing a lane-changing algorithm that would dynamically change the target pixel value in order to move from one lane to another.
+
The complete set of configuration values and their defaults can be found in donkeycar/templates/cfg_cv_control.py and is copied here for convenience.
+# configure which part is used as the autopilot - change to use your own autopilot
+CV_CONTROLLER_MODULE = "donkeycar.parts.line_follower"
+CV_CONTROLLER_CLASS = "LineFollower"
+CV_CONTROLLER_INPUTS = ['cam/image_array']
+CV_CONTROLLER_OUTPUTS = ['pilot/steering', 'pilot/throttle', 'cv/image_array']
+CV_CONTROLLER_CONDITION = "run_pilot"
+
+# LineFollower - line color and detection area
+SCAN_Y = 120 # num pixels from the top to start horiz scan
+SCAN_HEIGHT = 20 # num pixels high to grab from horiz scan
+COLOR_THRESHOLD_LOW = (0, 50, 50) # HSV dark yellow (opencv HSV hue value is 0..179, saturation and value are both 0..255)
+COLOR_THRESHOLD_HIGH = (50, 255, 255) # HSV light yellow (opencv HSV hue value is 0..179, saturation and value are both 0..255)
+
+# LineFollower - target (expected) line position and detection thresholds
+TARGET_PIXEL = None # In not None, then this is the expected horizontal position in pixels of the yellow line.
+ # If None, then detect the position yellow line at startup;
+ # so this assumes you have positioned the car prior to starting.
+TARGET_THRESHOLD = 10 # number of pixels from TARGET_PIXEL that vehicle must be pointing
+ # before a steering change will be made; this prevents algorithm
+ # from being too twitchy when it is on or near the line.
+CONFIDENCE_THRESHOLD = (1 / IMAGE_W) / 3 # The fraction of total sampled pixels that must be yellow in the sample slice.
+ # The sample slice will have SCAN_HEIGHT pixels and the total number
+ # of sampled pixels is IMAGE_W x SCAN_HEIGHT, so if you want to make sure
+ # that all the pixels in the sample slice are yellow, then the confidence
+ # threshold should be SCAN_HEIGHT / (IMAGE_W x SCAN_HEIGHT) or (1 / IMAGE_W).
+ # If you keep getting `No line detected` logs in the console then you
+ # may want to lower the threshold.
+
+# LineFollower - throttle step controller; increase throttle on straights, descrease on turns
+THROTTLE_MAX = 0.3 # maximum throttle value the controller will produce
+THROTTLE_MIN = 0.15 # minimum throttle value the controller will produce
+THROTTLE_INITIAL = THROTTLE_MIN # initial throttle value
+THROTTLE_STEP = 0.05 # how much to change throttle when off the line
+
+# These three PID constants are crucial to the way the car drives. If you are tuning them
+# start by setting the others zero and focus on first Kp, then Kd, and then Ki.
+PID_P = -0.01 # proportional mult for PID path follower
+PID_I = 0.000 # integral mult for PID path follower
+PID_D = -0.0001 # differential mult for PID path follower
+
+OVERLAY_IMAGE = True # True to draw computer vision overlay on camera image in web ui
+ # NOTE: this does not affect what is saved to the data
+
+
+It is very common to use a Proportional Integral Derivative (PID) controller to control throttle and/or steering in a wheeled robot. For example, the Path Follow autopilot uses a PID algorithm to modify steering based on how far away from the desired path the robot is. In the Computer Vision template, the built-in Line Follower algorithm uses a PID in a similar way; the line follow algorithm outputs a value that is proportional to how far the car is from the center line and whose sign indicates which side of the line it is on. The PID controller uses the magnitude and sign of the distance from the center line to calculate a steering value that will move the car towards the center line.
+++++The path_follow autopilot also uses a PID controller. There is a good description of how to tune a controller for driving at Determining PID Coefficients
+
You can use the CV_CONTROLLER_*
configuration values to point to a python file and class that implements your own computer vision autopilot part. Your autopilot class must conform to the donkeycar part standard. You can also determine the name of the input values, output values and run_condition. The default configuration values point the the included LineFollower part. At a minimum computer vistion autpilot part takes the camera image as an input and outputs the autopilot's throttle and steering values.
Let's create a simple custom computer vision part. It won't be much of an autopilot because it will just output a constant throttle and steering value and an image that counts that frames.
+A computer vision is a donkeycar part, so at a minimum it must be a Python class with a run(self)
method. An autpilot needs a little more than that, which we will see, but here is a minimal structure;
import cv2
+import numpy as np
+from simple_pid import PID
+import logging
+
+logger = logging.getLogger(__name__)
+
+
+class MockCvPilot:
+ def __init__(self, pid, cfg):
+ # initialize instance properties
+ pass
+
+ def run(self, img):
+ # use img to determine a steering and throttle value
+ return 0, 0, None # steering, throttle, image
+
+The constructor, __init__(self, pid, cfg)
takes a PID controller instance and the vehicle configuration properties. It is very common for autopilots to use a PID controller, so the framework provides one. Your autopilot may have values that you want to adjust to tune the algorithm; you should put those values in the myconfig.py
configuration file, then retrieve them in the constructor. In our MockCvPilot
we want to know if the user wants to see the telemetry image or just the camera image. We do the same thing in the built-in LineFollower
autopilot part, so we can just re-use that configuration value, OVERLAY_IMAGE
, in our autpilot. We can add that in our constructor;
def __init__(self, pid, cfg):
+ self.pid_st = pid
+ self.overlay_image = cfg.OVERLAY_IMAGE
+ self.counter = 0
+
+The run(self, img)
method is called each time through the loop. This is where you will interpret the image that is passed and determine a steering and throttle value that the car should use. The Computer Vision template also allows for showing an image in the web ui that is different in autopilot mode; typically you would add telemetry information to the camera image that is passed to run; such as the new steering and throttle values and perhaps a other alterations to the image so the user can better understand how the algorithm is working. For instance, if your algorithm did edge detection using the Canny algorithm, then you might want to show the processed image with the edges. So the minimal autopilot part returns a tuple of (steering, throttle, image).
To keep things simple, the MockCvPilot won't actually predict a steering and throttle, it will just return zero for each. However it will maintain a counter and display that in the telemetry image. We can see that in the run()
method.
def run(self, cam_img):
+ if cam_img is None:
+ return 0, 0, None
+
+ self.counter += 1
+
+ # show some diagnostics
+ if self.overlay_image:
+ # draw onto a COPY of the image so we don't alter the original
+ cam_img = self.overlay_display(np.copy(cam_img))
+
+ return self.steering, self.throttle, cam_img
+
+There are a couple of things to note here:
+run()
method is protected against an empty camera image - this can happen, espcially during startup. So in this case we stop the car.OVERLAY_IMAGE
, that we copied into self.overlay_image
is True
. If it is not True
then we just pass throught the original camera image.We put the logic that draws the telemetry image into it's own method so keep the both it and the run()
method clean and cohesive. Also because we run()
method has made a defensive copy of the original image, the method can do anything it wants to the image; even overwrite it completely. In our case we just draw some text on it to show the steering, throttle and counter values. We know the steering and throttle values will be zero in our mock autopilot, but it is instructive to show how you might display them. In this case we are showing them as text, but you might prefer to show them as bars, like we do in the webui, or some other visualization. This is the display method we use in our mock autopilot;
def overlay_display(self, img):
+ display_str = []
+ display_str.append(f"STEERING:{self.steering:.1f}")
+ display_str.append(f"THROTTLE:{self.throttle:.2f}")
+ display_str.append(f"COUNTER:{self.counter}")
+
+ lineheight = 25
+ y = lineheight
+ x = lineheight
+ for s in display_str:
+ cv2.putText(img, s, color=(0, 0, 0), org=(x ,y), fontFace=cv2.FONT_HERSHEY_SIMPLEX, fontScale=1, thickness=3)
+ cv2.putText(img, s, color=(0, 255, 0), org=(x ,y), fontFace=cv2.FONT_HERSHEY_SIMPLEX, fontScale=1, thickness=1)
+ y += lineheight
+
+ return img
+
+There are a couple of things to note: +- We organize the text as an array of strings; that makes it easy to process the line when we are drawing the text. You might even want to have a separate method to create this list of text string and possibly pass them into the display() methed if that simplies the display method or makes it more versatile (it can do both). +- We draw the text twice, once with a thick black stroke and then again with a thinner green stroke. This creates green text with a black outline; this makes it easier to read on an unpredictable background.
+Here is the complete custom computer vision autopilot part:
+import cv2
+import numpy as np
+from simple_pid import PID
+import logging
+
+logger = logging.getLogger(__name__)
+
+
+class MockCvPilot:
+ '''
+ OpenCV based MOCK controller; just draws a counter and
+ returns 0 for thottle and steering.
+
+ :param pid: a PID controller that can be used to estimate steering and/or throttle
+ :param cfg: the vehicle configuration properties
+ '''
+ def __init__(self, pid, cfg):
+ self.pid_st = pid
+ self.overlay_image = cfg.OVERLAY_IMAGE
+ self.steering = 0
+ self.throttle = 0
+ self.counter = 0
+
+
+ def run(self, cam_img):
+ '''
+ main runloop of the CV controller.
+
+ :param cam_img: the camerate image, an RGB numpy array
+ :return: tuple of steering, throttle, and the telemetry image.
+
+ If overlay_image is True, then the output image
+ includes an overlay that shows how the
+ algorithm is working; otherwise the image
+ is just passed-through untouched.
+ '''
+ if cam_img is None:
+ return 0, 0, None
+
+ self.counter += 1
+
+ # show some diagnostics
+ if self.overlay_image:
+ # draw onto a COPY of the image so we don't alter the original
+ cam_img = self.overlay_display(np.copy(cam_img))
+
+ return self.steering, self.throttle, cam_img
+
+ def overlay_display(self, img):
+ '''
+ draw on top the given image.
+ show some values we are using for control
+
+ :param img: the image to draw on as a numpy array
+ :return: the image with overlay drawn
+ '''
+ # some text to show on the overlay
+ display_str = []
+ display_str.append(f"STEERING:{self.steering:.1f}")
+ display_str.append(f"THROTTLE:{self.throttle:.2f}")
+ display_str.append(f"COUNTER:{self.counter}")
+
+ lineheight = 25
+ y = lineheight
+ x = lineheight
+ for s in display_str:
+ # green text with black outline so it shows up on any background
+ cv2.putText(img, s, color=(0, 0, 0), org=(x ,y), fontFace=cv2.FONT_HERSHEY_SIMPLEX, fontScale=1, thickness=3)
+ cv2.putText(img, s, color=(0, 255, 0), org=(x ,y), fontFace=cv2.FONT_HERSHEY_SIMPLEX, fontScale=1, thickness=1)
+ y += lineheight
+
+ return img
+
+To use the custom part, we must modify the myconfig.py file in the mycar folder to locate the python file and the class within it and to specify the inputs, outputs and run_condition that should be used when adding the part to the vehicle loop:
+# # configure which part is used as the autopilot - change to use your own autopilot
+CV_CONTROLLER_MODULE = "my_cv_pilot"
+CV_CONTROLLER_CLASS = "MockCvPilot"
+CV_CONTROLLER_INPUTS = ['cam/image_array']
+CV_CONTROLLER_OUTPUTS = ['pilot/steering', 'pilot/throttle', 'cv/image_array']
+CV_CONTROLLER_CONDITION = "run_pilot"
+
+CV_CONTROLLER_MODULE
is the package path to the my_cv_autpilot.py
file. It is generally is convenient to have this in the mycar folder and this is what we have done here. However, if you are developing this in your own repository, then if you are a Mac or Linux machine you can create a symbolic link to the file or the folder in which the file or files are.
CV_CONTROLLER_CLASS
is the name of the part's class in the python file to which CV_CONTROLLER_MODULE
points. In our case this is MockCvPilot
.
CV_CONTROLLER_INPUTS
is an array of the named inputs to the part that are passed when the part is added to the vehicle loop. For a computer vision autopilot the image is the minimum required. However you can pass any named values in the vehicle's memory. These correspond in a one-to-one fashion to the arguments (ignore the self argument) to the autopilot's run() method. So our mock example expects only an image run(self, cam_img)
and we only declare an image in the inputs, ['cam/image_array']
.
CV_CONTROLLER_OUTPUTS
is an array of named outputs to the part that are passed when the part is added to vehicle loop. These correspond to the return values from the autopilot's run()
. This is an autopilot, so we return a steering value and a throttle value. We also produce a new image with telemetry information drawn on it. So our mock autopilot returns return self.steering, self.throttle, cam_img
which corresponds to the declared output values, ['pilot/steering', 'pilot/throttle', 'cv/image_array']
.
CV_CONTROLLER_CONDITION
is the named value that decides if the autopilot part will run or not run. If you always want it to run, then pass None
, otherwise this should be the name of a boolean value; when it is True
the part's run()
method will be called; when it if False
the run()
method is not called. The templates maintain such a boolean value named "run_pilot"
, so we use that.
Now that you understand the structure of an autopilot part, it is worth reviewing the pseudocode in The Line Follower section above and compare that to the actual implementation. The python file is located at https://github.com/autorope/donkeycar/blob/main/donkeycar/parts/line_follower.py and is copied below. In particular:
+ def get_i_color(self, cam_img):
+ # take a horizontal slice of the image
+ iSlice = self.scan_y
+ scan_line = cam_img[iSlice : iSlice + self.scan_height, :, :]
+
+ # convert to HSV color space
+ img_hsv = cv2.cvtColor(scan_line, cv2.COLOR_RGB2HSV)
+
+ # make a mask of the colors in our range we are looking for
+ mask = cv2.inRange(img_hsv, self.color_thr_low, self.color_thr_hi)
+
+ # which index of the range has the highest amount of yellow?
+ hist = np.sum(mask, axis=0)
+ max_yellow = np.argmax(hist)
+
+ return max_yellow, hist[max_yellow], mask
+
+ max_yellow, confidence, mask = self.get_i_color(cam_img)
+ if self.target_pixel is None:
+ self.target_pixel = max_yellow
+
+run()
method with the TARGET_PIXEL
value. if self.pid_st.setpoint != self.target_pixel:
+ # this is the target of our steering PID controller
+ self.pid_st.setpoint = self.target_pixel
+
+ if confidence >= self.confidence_threshold:
+ # invoke the controller with the current yellow line position
+ # get the new steering value as it chases the ideal target_value
+ self.steering = self.pid_st(max_yellow)
+
+ if abs(max_yellow - self.target_pixel) > self.target_threshold:
+ # we will be turning, so slow down
+ if self.throttle > self.throttle_min:
+ self.throttle -= self.delta_th
+
+ else:
+ # we are going straight, so speed up
+ if self.throttle < self.throttle_max:
+ self.throttle += self.delta_th
+
+
+Here is the complete source to the LineFollower
part.
import cv2
+import numpy as np
+from simple_pid import PID
+import logging
+
+logger = logging.getLogger(__name__)
+
+
+class LineFollower:
+ '''
+ OpenCV based controller
+ This controller takes a horizontal slice of the image at a set Y coordinate.
+ Then it converts to HSV and does a color thresh hold to find the yellow pixels.
+ It does a histogram to find the pixel of maximum yellow. Then is uses that iPxel
+ to guid a PID controller which seeks to maintain the max yellow at the same point
+ in the image.
+ '''
+ def __init__(self, pid, cfg):
+ self.overlay_image = cfg.OVERLAY_IMAGE
+ self.scan_y = cfg.SCAN_Y # num pixels from the top to start horiz scan
+ self.scan_height = cfg.SCAN_HEIGHT # num pixels high to grab from horiz scan
+ self.color_thr_low = np.asarray(cfg.COLOR_THRESHOLD_LOW) # hsv dark yellow
+ self.color_thr_hi = np.asarray(cfg.COLOR_THRESHOLD_HIGH) # hsv light yellow
+ self.target_pixel = cfg.TARGET_PIXEL # of the N slots above, which is the ideal relationship target
+ self.target_threshold = cfg.TARGET_THRESHOLD # minimum distance from target_pixel before a steering change is made.
+ self.confidence_threshold = cfg.CONFIDENCE_THRESHOLD # percentage of yellow pixels that must be in target_pixel slice
+ self.steering = 0.0 # from -1 to 1
+ self.throttle = cfg.THROTTLE_INITIAL # from -1 to 1
+ self.delta_th = cfg.THROTTLE_STEP # how much to change throttle when off
+ self.throttle_max = cfg.THROTTLE_MAX
+ self.throttle_min = cfg.THROTTLE_MIN
+
+ self.pid_st = pid
+
+
+ def get_i_color(self, cam_img):
+ '''
+ get the horizontal index of the color at the given slice of the image
+ input: cam_image, an RGB numpy array
+ output: index of max color, value of cumulative color at that index, and mask of pixels in range
+ '''
+ # take a horizontal slice of the image
+ iSlice = self.scan_y
+ scan_line = cam_img[iSlice : iSlice + self.scan_height, :, :]
+
+ # convert to HSV color space
+ img_hsv = cv2.cvtColor(scan_line, cv2.COLOR_RGB2HSV)
+
+ # make a mask of the colors in our range we are looking for
+ mask = cv2.inRange(img_hsv, self.color_thr_low, self.color_thr_hi)
+
+ # which index of the range has the highest amount of yellow?
+ hist = np.sum(mask, axis=0)
+ max_yellow = np.argmax(hist)
+
+ return max_yellow, hist[max_yellow], mask
+
+
+ def run(self, cam_img):
+ '''
+ main runloop of the CV controller
+ input: cam_image, an RGB numpy array
+ output: steering, throttle, and the image.
+ If overlay_image is True, then the output image
+ includes and overlay that shows how the
+ algorithm is working; otherwise the image
+ is just passed-through untouched.
+ '''
+ if cam_img is None:
+ return 0, 0, False, None
+
+ max_yellow, confidence, mask = self.get_i_color(cam_img)
+ conf_thresh = 0.001
+
+ if self.target_pixel is None:
+ # Use the first run of get_i_color to set our relationship with the yellow line.
+ # You could optionally init the target_pixel with the desired value.
+ self.target_pixel = max_yellow
+ logger.info(f"Automatically chosen line position = {self.target_pixel}")
+
+ if self.pid_st.setpoint != self.target_pixel:
+ # this is the target of our steering PID controller
+ self.pid_st.setpoint = self.target_pixel
+
+ if confidence >= self.confidence_threshold:
+ # invoke the controller with the current yellow line position
+ # get the new steering value as it chases the ideal target_value
+ self.steering = self.pid_st(max_yellow)
+
+ # slow down linearly when away from ideal, and speed up when close
+ if abs(max_yellow - self.target_pixel) > self.target_threshold:
+ # we will be turning, so slow down
+ if self.throttle > self.throttle_min:
+ self.throttle -= self.delta_th
+ if self.throttle < self.throttle_min:
+ self.throttle = self.throttle_min
+ else:
+ # we are going straight, so speed up
+ if self.throttle < self.throttle_max:
+ self.throttle += self.delta_th
+ if self.throttle > self.throttle_max:
+ self.throttle = self.throttle_max
+ else:
+ logger.info(f"No line detected: confidence {confidence} < {self.confidence_threshold}")
+
+ # show some diagnostics
+ if self.overlay_image:
+ cam_img = self.overlay_display(cam_img, mask, max_yellow, confidence)
+
+ return self.steering, self.throttle, cam_img
+
+ def overlay_display(self, cam_img, mask, max_yellow, confidense):
+ '''
+ composite mask on top the original image.
+ show some values we are using for control
+ '''
+
+ mask_exp = np.stack((mask, ) * 3, axis=-1)
+ iSlice = self.scan_y
+ img = np.copy(cam_img)
+ img[iSlice : iSlice + self.scan_height, :, :] = mask_exp
+ # img = cv2.cvtColor(img, cv2.COLOR_RGB2BGR)
+
+ display_str = []
+ display_str.append("STEERING:{:.1f}".format(self.steering))
+ display_str.append("THROTTLE:{:.2f}".format(self.throttle))
+ display_str.append("I YELLOW:{:d}".format(max_yellow))
+ display_str.append("CONF:{:.2f}".format(confidense))
+
+ y = 10
+ x = 10
+
+ for s in display_str:
+ cv2.putText(img, s, color=(0, 0, 0), org=(x ,y), fontFace=cv2.FONT_HERSHEY_SIMPLEX, fontScale=0.4)
+ y += 10
+
+ return img
+
+
+ If you are not already, please ssh into your vehicle.
+Create a set of files to control your Donkey with this command:
+donkey createcar --path ~/mycar
+
+That creates a car using the default deep learning template. You can also create a car the uses the gps path follow template;
+donkey createcar --template=path_follow --path ~/mycar
+
+You can also create a car the uses the computer vision template;
+donkey createcar --template=cv_control --path ~/mycar
+
++++++
mycar
is not a special name; you can name your car folder anything you want.
See also more information on createcar.
+Look at myconfig.py in your newly created directory, ~/mycar
+cd ~/mycar
+nano myconfig.py
+
+Each line has a comment mark. The commented text shows the default value. When you want to make an edit to over-write the default, uncomment the line by removing the # and any spaces before the first character of the option.
+example:
+# STEERING_LEFT_PWM = 460
+
+becomes:
+STEERING_LEFT_PWM = 500
+
+when edited. You will adjust these later in the calibrate section.
+If you are using a PCA9685 servo driver board, make sure you can see it on I2C.
+Jetson Nano:
+sudo usermod -aG i2c $USER
+sudo reboot
+
+After a reboot, then try:
+sudo i2cdetect -r -y 1
+
+Raspberry Pi:
+sudo apt-get install -y i2c-tools
+sudo i2cdetect -y 1
+
+This should show you a grid of addresses like:
+ 0 1 2 3 4 5 6 7 8 9 a b c d e f
+00: -- -- -- -- -- -- -- -- -- -- -- -- --
+10: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+20: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+30: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+40: 40 -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+50: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
+70: 70 -- -- -- -- -- -- --
+
+In this case, the 40 shows up as the address of our PCA9685 board. If this does not show up, then check your wiring to the board. On a pi, ensure I2C is enabled in menu of sudo raspi-config
(notice, it suggest reboot).
If you have assigned a non-standard address to your board, then adjust the address in the myconfig.py
under variable PCA9685_I2C_ADDR
. If your board is on another bus, then you can specify that with the PCA9685_I2C_BUSNUM
.
Jetson Nano: set PCA9685_I2C_BUSNUM = 1
in your myconfig.py . For the pi, this will be auto detected by the Adafruit library. But not on the Jetson Nano.
Set HAVE_SOMBRERO = True
in your myconfig.py if you have a sombrero board.
Set HAVE_ROBOHAT = True
in your myconfig.py if you have a Robo HAT MM1 board. Also set the following variables according to your setup. Most people will be using the below values, however, if you are using a Jetson Nano, please set MM1_SERIAL_PORT = '/dev/ttyTHS1'
#ROBOHAT MM1
+HAVE_ROBOHAT = True # set to true when using the Robo HAT MM1 from Robotics Masters. This will change to RC Control.
+MM1_STEERING_MID = 1500 # Adjust this value if your car cannot run in a straight line
+MM1_MAX_FORWARD = 2000 # Max throttle to go fowrward. The bigger the faster
+MM1_STOPPED_PWM = 1500
+MM1_MAX_REVERSE = 1000 # Max throttle to go reverse. The smaller the faster
+MM1_SHOW_STEERING_VALUE = False
+# Serial port
+# -- Default Pi: '/dev/ttyS0'
+# -- Jetson Nano: '/dev/ttyTHS1'
+# -- Google coral: '/dev/ttymxc0'
+# -- Windows: 'COM3', Arduino: '/dev/ttyACM0'
+# -- MacOS/Linux:please use 'ls /dev/tty.*' to find the correct serial port for mm1
+# eg.'/dev/tty.usbmodemXXXXXX' and replace the port accordingly
+MM1_SERIAL_PORT = '/dev/ttyS0' # Serial Port for reading and sending MM1 data (raspberry pi default)
+
+# adjust controller type as Robohat MM1
+CONTROLLER_TYPE='MM1'
+# adjust drive train for web interface
+DRIVE_TRAIN_TYPE = 'MM1'
+
+The Robo HAT MM1 uses a RC Controller and CircuitPython script to drive the car during training. You must put the CircuitPython script onto the Robo HAT MM1 with your computer before you can continue.
+You may need to enable the hardware serial port on your Raspberry Pi. On your Raspberry Pi...
+sudo raspi-config
If you would like additional hardware or software support with Robo HAT MM1, there are a few guides published on Hackster.io. They are listed below.
+ + + +If you plan to use a joystick, take a side track over to here.
+If you are using the default deep learning template or the computer vision template then you will need a camera. By default myconfig.py assumes a RaspberryPi camera. You can change this by editing the CAMERA_TYPE
value in the myconfig.py file in your ~/mycar
folder.
If you are using the gps path follow template then you do not need, and may not want, a camera. In this case you can change the camera type to mock; CAMERA_TYPE = "MOCK"
.
See Cameras for details on the various cameras and configuration.
+Make all config changes to myconfig.py and they will be preserved through an update. When changes occur that you would like to get, you can pull the latest code, then issue a:
+cd projects/donkeycar
+git pull
+donkey createcar --path ~/mycar --overwrite
+
+If you created a car with the gps path follow template then remember to include the --template argument;
+donkey createcar --template=path_follow --path ~/mycar --overwrite
+
+Your ~/mycar/manage.py, ~/mycar/config.py and other files will change with this operation, but myconfig.py will not be touched. Your data and models dirs will not be touched.
+The purpose of providing a sample dataset and pre-trained models here is to help beginners to get started easier. You can download the dataset and use it to train together with your own dataset.
+If you have a dataset you want to contribute, please contact us on Discord #dataset channel, or raise a PR on donkey_datasets. Thank you.
+Now available on donkey_datasets. We plan to grow the repository of pre-trained models.
+ +Robocar Controller is a mobile app designed to provide a “commandless” user experience to get started with the Donkey Car.
+ +Please refer to the quick start guide here.
+++If you do not want to use the prebuilt image then you can install the server component onto your Donkey Car manually. See Optional Manual Installation below.
+
The car will become a hotspot when there is no known Wifi network to connect. After connecting your phone to this hotspot, you can use the app to configure the car to join the Wifi network you want.
+Once your car connects to the same network as your phone, the app will scan the whole network to discover it. The app will also show you the IP address of the car in case you want to connect to it via SSH.
+ +Sometimes it is quite annoying if the car goes too fast or does not run in a straight line. The calibration UI assists you to find the right settings so that your car could be properly calibrated. With the enhanced calibration function, the change will take place in real time and you could observe the change immediately.
+ +The virtual joystick offers a quick way to test drive the car if you don't have a physical gamepad controller. It also streams the video captured from the camera in real time. You can just look at the screen and start driving.
+ +The app presents a drive summary with histogram, the size and the number of images you have collected. The histogram is generated automatically by calling the tubhist
function in the Donkey car software.
The app shows all the data(tubs) and the metadata you have collected on the Pi. The metadata includes number of images, size of the tub, the resolutions, the histogram and the location. The app will make use of the donkey makemovie command to generate a video so you can review how the data look like.
+ +Free GPU training is available to user who use the app. You can train a model by selecting the data(tubs) you wish to train. The data will be uploaded to our server to start the training process. Once the training is completed, the app will show you the training loss and accuracy graph. At the same time, the app will download the model to your car and you can test the model right away.
+Note: We keep the data and models for a period of time. After that, we will delete it from our storage.
+ +We are using AWS g4dn.xlarge instance to train the model. It feautres NVIDIA T4 GPU and up to 16GB GPU memory. Increase the batch size to 256 or more to fully utilize the powerful GPU.
+N.B.: To protect our equipment from being abused, we have the following rules to use the training service.
+The app will list all models inside the Pi, no matter it is generated from the training function or just a model copied to the Pi. You can start the autopilot mode using a similar UI as the Drive UI.
+ +The Doneky car software comes with a vast of configuration that you can experiment. We have included some of the popular options that you may want to change.
+If you are using MM1, the app shows you the current battery level in percentage. We have also added an OS tweak that if battery level fall below 7V, the system will shutdown automatically.
+If you encountered a problem, please file an issue on this github project.
+If you can not or do not want to use the prebuild SD image for you Donkey Car, then you can install the server component onto your Donkey car manually. +Donkey Car console is a management software of the donkey car that provides a rest-based API to support Donkey Car mobile app.
+++Note This software currently supports RaspberryPi 4B only.
+
git clone https://github.com/robocarstore/donkeycar-console
+sudo mv donkeycar-console /opt
+cd /opt/donkeycar-console
+
+pip install -r requirements/production.txt
+
+python manage.py migrate
+
+python manage.py runserver 0.0.0.0:8000
+
+Go to http://your_pi_ip:8000/vehicle/status. If it returns something without error, it works.
+sudo ln -s gunicorn.service /etc/systemd/system/gunicorn.service
+
+Make sure your phone is connected to the same network as your Pi (if it won't connect, try turning off your cell data). Fire up the mobile app and you can search your car using the mobile app.
+We would love to call the app Donkeycar Controller but Apple does not allow us to do so. We are working with Adam to submit a proof to Apple that we can use the Donkeycar trademark in our app. In the meanwhile, we will be using the name Robocar Controller.
+This app is developed by Robocar Store. If you plan to use this app to make money, please follow the Donkey Car guideline and send an email to Robocar Store.
+ +The Donkey Gym project is a OpenAI gym wrapper around the Self Driving Sandbox donkey simulator (sdsandbox
). When building the sim from source, checkout the donkey
branch of the sdsandbox
project.
The simulator is built on the the Unity game platform, uses their internal physics and graphics, and connects to a donkey Python process to use our trained model to control the simulated Donkey.
+Here's some videos to help you through the installation.
+Linux: +https://youtu.be/J6Ll5Obtuxk
+Windows: +https://youtu.be/wqQMmHVT8qw
+There are many ways to use the simulator, depending on your goals. You can use the simulator to get to know and use the standard Donkeycar drive/train/test cycle by treating it as virtual hardware. You will collect data, drive, and train using the same commands as if you were using a real robot. We will walk through that use-case first.
++
cd ~/projects
+git clone https://github.com/tawnkramer/gym-donkeycar
+cd gym-donkeycar
+conda activate donkey
+pip install -e .[gym-donkeycar]
+
+donkey createcar --path ~/mysim
+cd ~/mysim
+
+<user-name>
and the other parts of the path:DONKEY_GYM = True
+DONKEY_SIM_PATH = "/home/<user-name>/projects/DonkeySimLinux/donkey_sim.x86_64"
+DONKEY_GYM_ENV_NAME = "donkey-generated-track-v0"
+
+++Note: your path to the executable will vary depending on platform and user. + Windows: DonkeySimWin/donkey_sim.exe + Mac OS: DonkeySimMac/donkey_sim.app/Contents/MacOS/donkey_sim + Linux: DonkeySimLinux/donkey_sim.x86_64
+
You may use all the normal commands to manage.py at this point. Such as:
+python manage.py drive
+
+This should start the simulator and connect to it automatically. By default you will have a web interface to control the donkey. Navigate to http://localhost:8887/drive to see control page.
+On Ubuntu Linux only, you may plug in your joystick of choice.
+If it mounts as /dev/input/js0
then there's a good chance it will work.
+Modify myconfig.py to indicate your joystick model and use the --js
arg to run.
python manage.py drive --js
+
+As you drive, this will create a tub of records in your data dir as usual.
+You will not need to rsync your data, as it was recorded and resides locally. You can train as usual:
+donkey train --tub ./data --model models/mypilot.h5
+
+You can use the model as usual:
+python manage.py drive --model models/mypilot.h5
+
+Then navigate to web control page. Set Mode and Pilot
to Local Pilot(d)
. The car should start driving.
Here's some sample driving data to get you started. Download this and unpack it into your data dir. This should train to a slow but stable driver.
+Here's some info on the api to talk to the sim server. Make a TCP client and connect to port 9091 on whichever host the sim is running. The server sends and receives UTF-8 encoded JSON packets. Each message must have a "msg_type" field. The sim will end all JSON packets with a newline character for termination. You don't have to end each packet with a newline when sending to the server. But if it gets too many messages too quickly it may have troubles. Check the player log file for JSON parse errors if you are having troubles.
+Client=>Sim. Ask for the version of the protocol. Will help know when changes are made to these messages.
+Fields: None
+Example:
+ {
+ "msg_type" : "get_protocol_version"
+ }
+
+Sim=>Client. Reply for the version of the protocol. Currently at version 2.
+Fields:
+Example:
+ {
+ "msg_type" : "protocol_version",
+ "version" : "2",
+ }
+
+Sim=>Client. When the Menu scene is finished loading this will be sent. After this point, the sim can honor the Scene Loading message. (Menu only)
+Fields: None
+Example:
+ {
+ "msg_type" : "scene_selection_ready"
+ }
+
+Client=>Sim. Ask names of the scene you can load. (Menu only)
+Fields: None
+Example:
+ {
+ "msg_type" : "get_scene_names"
+ }
+
+Sim=>Client. Sim will reply with list of scene names.
+Fields:
+Example:
+ {
+ "msg_type" : "scene_names"
+ "scene_names" : [ "generated_road", "warehouse", "sparkfun_avc". "generated_track" ]
+ }
+
+Client=>Sim. Asks the sim to load one of the scenes from the Menu screen. (Menu only)
+Fields:
+scene_name : generated_road | warehouse | sparkfun_avc | generated_track ( or whatever list the sim returns from get_scene_names)
+Example:
+ {
+ "msg_type" : "load_scene",
+ "scene_name" : "generated_track"
+ }
+
+Sim=>Client. Once scene is loaded, in reply, you will get a:
+ {
+ "msg_type" : "scene_loaded"
+ }
+
+Sim=>Client. Once the sim finishes loading your car, it sends this message. The car is loaded for you automatically once the scene is loaded with an active client. Or a client make a connection.
+Fields: None
+Example:
+ {
+ "msg_type" : "car_loaded"
+ }
+
+Client=>Sim. Once loaded, you may configure your car visual details (scene only)
+Fields:
+Example:
+ {
+ "msg_type" : "car_config",
+ "body_style" : "car01",
+ "body_r" : "128",
+ "body_g" : "0",
+ "body_b" : "255",
+ "car_name" : "Your Name",
+ "font_size" : "100"
+ }
+
+Client=>Sim. Once the scene is loaded, you may configure your car camera sensor details
+Fields:
+Example:
+ {
+ "msg_type" : "cam_config",
+ "fov" : "150",
+ "fish_eye_x" : "1.0",
+ "fish_eye_y" : "1.0",
+ "img_w" : "255",
+ "img_h" : "255",
+ "img_d" : "1",
+ "img_enc" : "PNG",
+ "offset_x" : "0.0",
+ "offset_y" : "3.0",
+ "offset_z" : "0.0",
+ "rot_x" : "90.0"
+ }
+
+Note: +You can add an other camera by changing the msg_type to "cam_config_b"
+Client=>Sim. Control throttle and steering.
+Fields:
+Example:
+ {
+ "msg_type" : "control",
+ "steering" : "0.0",
+ "throttle" : "0.3",
+ "brake" : "0.0"
+ }
+
+Sim=>Client. The sim sends this message containing camera image and details about vehicle state. These come at a regular rate set in the sim. Usually about 20 HZ.
+Fields:
+Example:
+ {
+ "msg_type" : "telemetry",
+ "steering_angle" : "0.0",
+ "throttle" : "0.0",
+ "speed" : "1.0",
+ "image" : "0x123...",
+ "hit" : "None",
+ "pos_x" : "0.0",
+ "pos_y" : "0.0",
+ "pos_z" : "0.0",
+ "accel_x" : "0.0",
+ "accel_y" : "0.0",
+ "accel_z" : "0.0",
+ "gyro_x" : "0.0",
+ "gyro_y" : "0.0",
+ "gyro_z" : "0.0",
+ "gyro_w" : "0.0",
+ "pitch" : "0.0",
+ "roll" : "0.0",
+ "yaw" : "0.0",
+ "activeNode" : "5"
+ "totalNodes" : "26"
+ "cte" : "0.5"
+ }
+
+Client=>Sim. Return the car to the start point.
+Fields: None
+Example:
+ {
+ "msg_type" : "reset_car"
+ }
+
+Client=>Sim. Move the car to the given position (training only)
+Fields:
+Example:
+ {
+ "msg_type" : "set_position"
+ "pos_x" : "0.0",
+ "pos_y" : "0.0",
+ "pos_z" : "0.0"
+ }
+
+or:
+ {
+ "msg_type" : "set_position"
+ "pos_x" : "0.0",
+ "pos_y" : "0.0",
+ "pos_z" : "0.0",
+ "qx" : "0.0",
+ "qy" : "0.2",
+ "qz" : "0.0",
+ "qw" : "1.0"
+ }
+
+Client=>Sim. Ask for a node_position packet
+Fields:
+Example:
+ {
+ "msg_type": "node_position",
+ "index": "0"
+ }
+
+Sim=>Client. node_position packet (received after sending a node_position packet)
+Fields:
+Example:
+ {
+ "msg_type": "node_position",
+ "Qx": "0",
+ "Qy": "0",
+ "Qz": "0",
+ "Qw": "1",
+ "pos_x": "0",
+ "pos_y": "0",
+ "pos_z": "0"
+ }
+
+Client=>Sim. Leave the scene and return to the main menu screen.
+Fields: None
+Example:
+ {
+ "msg_type" : "exit_scene"
+ }
+
+Client=>Sim. Close the sim executable. (Menu only)
+Fields: None
+Example:
+ {
+ "msg_type" : "quit_app"
+ }
+
+
+ Now that you're able to drive your car reliably you can use Keras to train a +neural network to drive like you. Here are the steps.
+Make sure you collect good, clean data. The neural network will learn what is in the data that it is train upon, so bad data results in a bad model. If your data has segments where you drive off the track, then there will be as aspect of the trained model that will reflect that. Ideally you would drive perfectly and no bad data would be recorded. However, that is not realistic, but there are ways to easily remove errors as they happen.
+A technique that is useful to avoid collecting bad data is to use the erase-records button on the controller when you make a mistake. So if you crash then immediately toggle recording to off and select the erase-records button to delete the last 100 records. It get's even easier if you use set AUTO_RECORD_ON_THROTTLE = True
in your myconfig.py
file; in that case recording will turn on when you apply throttle and will turn off when you stop applying throttle. So between that and the erase-records button you can record tens-of-thousands of records without having to go back and clean data with the donkey tubclean
command.
Beyond actual mistakes, the consistency in the data you record matters. The more variation there is in the data, the more data you will need to get a good model. The more consistent you are when you drive, then the more uniform the data will be and so the neural network will be able to correllate the inputs to the outputs more effectively. Consistency is particularly hard to get for throttle; it is very hard to replicate throttle exactly at each place on the track. One strategy (a strategy I use) is to find the maximum throttle you can maintain around the entire track or at least most of the track, then just use that; set JOYSTICK_MAX_THROTTLE
in your myconfig.py
file to that throttle, then you can put the pedal to the metal around most of the course when you are collecting data.
Start Recording
if using web controller or use AUTO_RECORD_ON_THROTTLE = True
as described above so the joystick will auto record with any non-zero throttle.Ctrl-c
in the ssh session for your car.donkey tubclean
to edit your data and remove the mistakes.Since the Raspberry Pi is not very powerful, we need to transfer the data +to a PC computer to train. The Jetson nano is more powerful, but still quite slow to train. If desired, skip this transfer step and train on the Nano.
+The easiest way to do the training on your desktop is by using the Donkey UI application. +
+If, however, you want to do the training with the command line, read on....
+In a new terminal session on your host PC use rsync to copy your cars +folder from the Raspberry Pi.
+rsync -rv --progress --partial pi@<your_pi_ip_address>:~/mycar/data/ ~/mycar/data/
+
+./data/*
to gather multiple manifests. For example, from your mycar folder on your host PC:~\mycar$ donkey train --tub ./data --model ./models/mypilot.h5
+
+You may specify more than one tub using a comma separated list --tub=foo/data,bar/data
or just leaving spaces like --tub foo/data bar/data
. See Train the Model
You can create different model types with the --type
argument during training. You may also choose to change the default model type in myconfig.py DEFAULT_MODEL_TYPE
. When specifying a new model type, be sure to provide that type when running the model, or using the model in other tools like plotting or profiling. For more information on the different model types, look here for Keras Parts. The model will be placed into the folder models/
. You can as well omit the --model
flag and the model name will be auto created using the pattern pilot_YY-MM-DD_N.h5
.
If you run with version >= 4.3.0, the model will be automatically created in tflite format for fast inferencing, generating a ./models/mypilot.tflite
file, too. Tflite creation can be suppressed, by setting CREATE_TF_LITE = False
in your myconfig.py
file. In addition, a tensorrt model is produced if you set CREATE_TENSOR_RT = True
, which is False
by default. That setting produces a ./models/mypilot.trt
file that should work on all platforms. On RPi, the tflite model will be the fastest.
++Note: There was a regression in version 4.2 where you only had to provide the model name in the model argument, like
+--model mypilot.h5
. This got resolved in version 4.2.1. Please update to that version.
Image Augmentation With version >= 4.3.0 you also have access to image augmentations for training. Image augmentation is a technique where data, in this case images, is changed (augmented) to create variation in the data. The purpose is two-fold. First it can help extend your data when you don't have a lot of data. Second it can create a model that is more resilient to variations in the data at inference time. In our case, we want to handle various lighting conditions. Currently supported are AUGMENTATIONS = ['MULTIPLY', 'BLUR']
in the settings which generate brightness modifications and apply a Gaussian blur. These can be used individually or together. Augmentations are only applied during training; they are not applied when driving on autopilot.
Image Transformation With version >= 4.3.0 you also have access to image transformations like cropping or trapezoidal masking. Cropping and masking are similar; both 'erase' pixels on the image. This is done to remove pixels that are not important and that may add unwanted detail that can make the model perform poorly under conditions where that unwanted detail is different. Cropping can erase pixels on the top, bottom, left and/or right of the image. Trapezoidal masking is a little more flexible in that it can mask pixels using a trapezoidal mask that can account for perspective in the image. To crop the image or apply a trapezoidal mask you can provide TRANSFORMATIONS = ['CROP']
or TRANSFORMATIONS = ['TRAPEZE']
. Generally you will use either cropping or trapezoidal masking but not both. Transformations must be applied in the same way in training and when driving on autopilot; make sure the transformation configuration is the same on your training machine and on your Donkey Car.
In previous step we managed to get a model trained on the data. Now is time to move the model back to Rasberry Pi, so we can use it for testing it if it will drive itself.
+Use rsync again to move your trained model pilot back to your car.
+rsync -rv --progress --partial ~/mycar/models/ pi@<your_ip_address>:~/mycar/models/
+
+Ensure to place car on the track so that it is ready to drive.
+Now you can start your car again and pass it your model to drive.
+python manage.py drive --model ~/mycar/models/mypilot.h5
+
+python manage.py drive --model ~/mycar/models/mypilot.tflite --type tflite_linear
+
+Read this for more information.
+a. User : As you guessed, this is where you are in control of both the steering and throttle control.
+b. Local Angle : Not too obvious, but this is where the trained model (mypilot from above) controls the steering. The Local refers to the trained model which is locally hosted on the raspberry-pi.
+c. Local Pilot : This is where the trained model (mypilot) assumes control of both the steering and the throttle. As of now, it's purportedly not very reliable.
+Be sure to also check out the Max Throttle and Throttle Mode options, and play around with a few settings. Can help with training quite a lot.
+Build a Simple Track : This isn't very well-documented, but the car should (theoretically) be able to train against any kind of track. To start off with, it might not be necessary to build a two-lane track with a striped center-lane. Try with a single lane with no center-line, or just a single strip that makes a circuit! At the least, you'll be able to do an end-to-end testing and verify that the software pipeline is all properly functional. Of course, as the next-step, you'll want to create a more standard track, and compete at a meetup nearest to you!
+Get help : Try to get some helping hands from a friend or two. Again, this helps immensely with building the track, because it is harder than it looks to build a two-line track on your own! Also, you can save on resources (and tapes) by using a ribbon instead of tapes. They'll still need a bit of tapes to hold them, but you can reuse them and they can be laid down with a lot less effort (Although the wind, if you're working outside, might make it difficult to lay them down initially).
+Make sure TRAIN_BEHAVIORS = True
in myconfig.py when training and when running on the robot.
Setup an RGB led on robot to indicate which state is active. Enable in config.py. Verify when running robot that L1 PS3 button changes state led indicator. (that's the left upper shoulder button)
+By default there are two states. If you like, adjust the number of states in bottom of config.py. Rename or change BEHAVIOR_LIST
to an arbitrary number of labels. Make sure same number of rgb colors in BEHAVIOR_LED_COLORS
. Make sure to reflect any changes to both PC and Robot.
Now for training: Activate any state with L1 shoulder button. Then drive as you wish the car to drive when in that state. Switch states and then transition to the new steady state behavior.
+For the two lane case. Drive 33% in one lane, 33% in the other, and 33% transitioning between them. It's important to trigger the state transition before changing lanes.
+Check the records in the data file. Open a .json. In addition to steering and throttle, you should also have some additional state information about your behavior vector and which was was activate on that frame. This is crucial to training correctly.
+Move data to PC and train as normal, ensuring TRAIN_BEHAVIORS = True
in myconfig.py on PC, otherwise extra state information will be ignored.
Move trained model back to robot. Now place the robot in the location of the initial state. Start the robot with the given model
+python manage.py drive --model=models/my_beh.h5 --type=behavior
+
+As it drives, you can now toggle states with L1 and see whether and how much it can replicate your steady state behaviors and transitions.
+Be sure to include quite a lot of example of transitions from one state to another. At least 50, but more like 100.
+ +We've taken the next step in DIY Robocars competitions. We are now hosting special events online! We welcome competitors from all over the world. Events will be scheduled by Chris Anderson and the Donkeycar maintainers. But this is by no means donkeycar only. Please read on, and we will provide two paths depending on whether you decide to use the donkeycar framework to race.
++
We will be broadcasting the race stream over Twitch. Check the event annoucement for the url. Race comepetitors will join a group Zoom chat event. Tawn will host the race server and share the stream over Zoom/Twitch. And we will see how things go.
+We are using the SDSandbox open source project as the racing sim. This creates a 3d environment using Unity game creation tool. It uses NVidia PhysX open source physics engine to simulate 4 wheeled vehicle dynamics. This sim also acts as a server, listening on TCP port 9091. This sends and receives JSON packets. More on the API later.
+We use an OpenAI GYM style wrapper to interface with the server. The project for this wrapper is gym-donkeycar.
+You can build the server from the source project above, or use pre-built binaries for Ubuntu, Mac, and Windows. This has been tested on Ubuntu 18.04, Mac 10.13, and Windows 10.
+If you are using the donkeycar framework to race, you can use follow the guide to setup the simulator. If visuals directions help out, checkout the Windows Sim Setup Screen-Cast on Youtube. Use this to practice before the race. When it comes time to race, modify your myconfig.py to have these two changes:
+DONKEY_SIM_PATH = "remote"
+SIM_HOST = "trainmydonkey.com"
+
+This racing server will not always be running. We will bring it up for testing events and on race day. We are aiming to have it up from 7pm-9pm Pacific every night a week before race day. If not up, ask on Discord and we will try to get things running.
+++Note: If you trained a donkey model, but wish to run it on a Jetson Nano or some platform where you are having troubles installing all the dependencies, here's a single script you can use to run without any donkeycar or gym-donkeycar dependencies. Just pass it the model file name, the host name, and the car name. And it will run as a client to the race sim.
+
If you would like to roll your own client, we have some python code to get you started.
+You will first want to download the sim pre-built binary for your platform. Extract that where you like.
+Then clone the gym-donkeycar python project and install. If you are using a virtual environment, don't forget to activate it first.
+git clone https://github.com/tawnkramer/gym-donkeycar
+pip install -e gym-donkeycar
+
+wget https://raw.githubusercontent.com/tawnkramer/sdsandbox/master/src/test_client.py
+
+then right click on test_client.py and choose "Save link as..." and choose a location on your PC.
+start up the simulator and let it get to the menu screen.
+python3 test_client.py
Checkout test_client.py to see what's going there. Class SimpleClient connects to the host of your choosing. Then it sends a load scene command depending on which course you want to try. It then sends some car visual configuration, and then some camera config information. Then it enters an update loop.
+You can try changing the num_clients variable to 2 or more clients. See how the sim can handle them.
+The test client will send random steering command for time_to_drive = 1.0 seconds. Then quit.
+During that time, the telemetry messages will come into SimpleClient::on_msg_recv. See them printed out for you. Also take a look at the 'test.png' that it writes to get a feel for what the camera looks like.
+There's some comments in there explaining the camera configuration in detail. If you have a custom camera setup, hopefully we can come close to matching it with these controls.
+When it's time to race, change the variable:
+host = "trainmydonkey.com"
+
+Be sure to enable controls to start the car on your command. We will likely be old school calling 3, 2, 1, GO! over video chat.
+There's a lot to learn. And come to Discord to get some help. Check out the #virtual-racing-league channel there.
+ +After you've calibrated your car you can start driving it.
+If you are not already, please ssh into your vehicle.
+++*** Put your car in a safe place where the wheels are off the ground ***.
+
This is the step were the car can take off.
+Open your car's folder and start your car.
+cd ~/mycar
+python manage.py drive
+
+This script will start the drive loop in your car which includes a part that
+is a web server for you to control your car. You can now control your car
+from a web browser at the URL: <your car's hostname.local>:8887
There are 3 ways to move the car using the web controller +- Device Tilt: +Select Device Tilt in the Control Mode section of the web controller. Then select User in the Mode section. You can then tilt your phone forward to increase throttle and tilt it side to side to turn the steering. +- Joystick: +Select Joystick in the Control Mode section of the web controller. Then select User in the Mode section. You can then touch and drag on the virtual joystick area that appears. Moving up increases throttle, moving down decreases or reverses. Moving left turns left, moving right turns right. Releasing you finger stops. +- Gamepad: +If you have a game controller attached (either by cable or by bluetooth) to the machine on which you are viewing the web ui, and that game controller is compatible with the HTML5 Gamepad API, then you can choose Gamepad in the Control Mode section of the web controller. Then select User in the Mode section. You can then drive the donkeycar using the game controller..
+space
: stop car and stop recordingr
: toggle recordingi
: increase throttlek
: decrease throttlej
: turn left l
: turn right If you don't have a joystick then you can skip to next section - train an autopilot.
+You may find that it helps to use a physical joystick device to control your vehicle.
+Check the Controllers section to read about setting up the bluetooth connection.
+cd ~/mycar
+python manage.py drive --js
+
+Optionally, if you want joystick use to be sticky and don't want to add the --js each time, modify your myconfig.py so that USE_JOYSTICK_AS_DEFAULT = True
+nano myconfig.py
+
+++Whenever the throttle is not zero, driving data will be recorded - as long as you are in User mode!
+
Install miniconda Python 3.9 64 bit
+Start Terminal
+Setup your donkey
conda env with:
conda create -n donkey python=3.9
+conda activate donkey
+
+Now there are two different installations possible. Very likely you will +want to do the user install. Then you will perform Step +User install. In case +you want to debug or edit the source code, you will need to do the more advanced +Developer install. But you can do only one.
+++Note: Only do User install or Developer install but not both!
+
As you have activated the new donkey
env already you simply type:
pip install donkeycar[pc]
+
+if you are using an Intel Mac or you type:
+pip install donkeycar[macos]
+
+if you are on Apple Silicon. This will install the latest release.
+Here you can choose which branch or tag you want to install, and you can +edit and/or debug the code, by downloading the source code from GitHub.
+Install git 64 bit +and change to a dir you would like to use as the head of your projects.
+mkdir projects
+cd projects
+git clone https://github.com/autorope/donkeycar
+cd donkeycar
+git checkout main
+pip install -e .[pc]
+
+Note: if you are using ZSH (you'll know if you are), you won't be able to
+run pip install -e .[pc]
. You'll need to escape the brackets and run
+pip install -e .\[pc\]
.
conda update -n base -c defaults conda
+conda env remove -n donkey
+
+Currently, there is no NVidia gpu support for +tensorflow on mac.
+donkey createcar --path ~/mycar
+
+++Note: After closing the Terminal, when you open it again, you will need to +type +
+conda activate donkey
+to re-enable the mappings to donkey specific +Python libraries
++Note : tested on Ubuntu 20.04 LTS, 22.04 LTS
+
Open the Terminal application.
+Install miniconda Python 3.9 64 bit.
+wget https://repo.anaconda.com/miniconda/Miniconda3-py39_23.3.1-0-Linux-x86_64.sh
+bash ./Miniconda3-py39_23.3.1-0-Linux-x86_64.sh
+
+Setup your donkey
conda env with:
conda create -n donkey python=3.9
+conda activate donkey
+
+Now there are two different installations possible. Very likely you will +want to do the user install. Then you will perform Step +User install. In case +you want to debug or edit the source code, you will need to do the more advanced +Developer install. But you can do only one.
+++Note: Only do User install or Developer install but not both!
+
As you have activated the new donkey
env already you simply type:
pip install donkeycar[pc]
+
+This will install the latest release.
+Here you can choose which branch or tag you want to install, and you can +edit and/or debug the code, by downloading the source code from GitHub.
+Create a project directory you would like to use as the
+head of your projects, change into it and download and install donkeycar
+from GitHub.
mkdir projects
+cd projects
+git clone https://github.com/autorope/donkeycar
+cd donkeycar
+git checkout main
+pip install -e .[pc]
+
+Note: if you are using ZSH (you'll know if you are), you won't be able to
+run pip install -e .[pc]
. You'll need to escape the brackets and run
+pip install -e .\[pc\]
.
conda update -n base -c defaults conda
+conda env remove -n donkey
+
+The newer version of Tensorflow is already built with GPU support. If you +have an Nvidia GPU, install Cuda 11 following instructions on Nivida's page +here
+If you have a Google Coral edge tpu, you may wish to compile models. You +will need to install the edgetpu_compiler exectutable. Follow their +instructions.
+If you have an NVidia card, you should update to the latest drivers and +install Cuda SDK.
+conda install cudatoolkit=11 -c pytorch
+
+You should replace <CUDA Version>
with your CUDA version. Any version
+above 10.0 should work. You can find out your CUDA version by running
+nvcc --version
or nvidia-smi
. (if those commands don't work, it means you
+don't already have them installed. Follow the directions given by that error
+to install them.) If the version given by these two commands don't match, go
+with the version given by nvidia-smi
.
donkey createcar --path ~/mycar
+
+++Note: After closing the Anaconda Prompt, when you open it again, you will need to +type
+conda activate donkey
to re-enable the mappings to donkey specific +Python libraries
Donkey Car used to support a native Windows installation but this has been +deprecated in favor of the WSL install.
+The Windows Subsystem for Linux (WSL) lets developers run a GNU/Linux environment -- including most command-line tools, utilities, and applications -- directly on Windows, unmodified, without the overhead of a traditional virtual machine or dualboot setup.
+Open the Ubuntu App and configure.
+Open the Ubuntu App to get a prompt window via Start Menu | Ubuntu
+Refresh list of packages and install pip and xclip:
+sudo apt-get update
+sudo apt install python3-pip
+sudo apt-get install libmtdev1 libgl1 xclip
+
+LD_PRELOAD=/usr/lib/x86_64-linux-gnu/libstdc++.so.6
to your .bashrc
and re-source it.
+ bash
+ source ~/.bashrc
At this point switch to the Ubuntu instructions and continue the setup there.
+If you use the Donkey UI with an NVIDIA graphics card and you see a blurred window it might be due to some settings on
+your PC. In the settings, switch the NVIDIA graphics card 3D rendering mode from
+let the running program decide the 3D rendering mode
+to let me decide on the 3D rendering mode: Quality
.
Donkeycar has components to install on a host PC. This can be a laptop, or desktop machine. The machine doesn't have to be powerful, but it will benefit from faster cpu, more ram, and an NVidia GPU. An SSD hard drive will greatly impact your training times.
+Donkeycar software components need to be installed on the robot platform of your choice. Raspberry Pi and Jetson Nano have setup docs. But it has been known to work on Jetson TX2, Friendly Arm SBC, or almost any Debian based SBC ( single board computer ).
+After install, you will create the Donkeycar application from a template. This contains code that is designed for you to customize for your particular case. Don't worry, we will get you started with some useful defaults.
+Next we will train the Donkeycar to drive on it's own based on your driving style! This uses a supervised learning technique often referred to as behavioral cloning.
+ + +This is not the only method for getting your Donkeycar to drive itself. But it requires the least amount of hardware and least technical knowledge. Then you can explore other techniques in this Ai mobile laboratory called Donkeycar!
+When controlling your Donkey via behavioral cloning, you will need to setup a host pc to train your machine learning model from the data collected on the robot. Choose a setup that matches your computer OS.
+This guide will help you to setup the software to run Donkeycar on your Raspberry Pi or Jetson Nano. Choose a setup that matches your SBC type. (SBC = single board computer)
+Setup RaspberryPi +
+Setup Jetson Nano +
+Read this for more information.
+Read this for more information.
+The path follow template is an alternative to the deep learning template. The deep learning template is great for an indoor track where lighting conditions and the details of the room can be controlled, but it can be more difficult to get working outside where lighting conditions are variable and things change in the environment. Outside we have access to GPS; the path_follow template allows you to record a path using a GPS receiver and then configure an autopilot that can follow that path.
+GPS positions are read from the GPS receiver over a serial port. We read these as NMEA sentences; a line oriented protocol that most GPS receivers use by default. The NMEA sentences include positions as latitude and longitude; we then convert those to a local coordinate system in meters.
+When we record a path, we save each (x, y) coordinate pair (each waypoint) we get from the GPS receiver into an array in memory. We can then save that to a CSV file which has one (x, y) coordinate pair per line. Later, we can read this csv file back into memory. Once we have our waypoints in memory, we can enter autopilot mode and follow those points.
+Similar to the deep learning template, we have 3 modes of operation:
+Path Follow Autopilot in Action
+Before we can record or follow a path, we need to create an application and do a little configuration.
+You can create a path follow application similarly to the how we create a deep learning application; we just tell it to use the path_follow template instead of the default template. First, make sure your donkeycar python environment is activated, then use the createcar command to create your application folder.
+donkey createcar --template=path_follow --path=~/mycar
+
+When updating to a new version of donkeycar, you will want to refresh your application folder. You can do this with the same command, but add --overwrite
so that it does not erase your myconfig.py file.
donkey createcar --template=path_follow --path=~/mycar --overwrite
+
+Again, like the deep learning template, we can change default configuration values by editing the myconfig.py file in the mycar folder you created with the createcar
command.
You will need to calibrate and configure the drivetrain as described in Calibrate your Car. If you have a game controller paired to your car, then you will want to configure it as described in Controllers.
+In myconfig.py, search for the 'gps' section. Make sure HAVE_GPS = True
is set. You will need to determine the serial port that the GPS receiver is connected to and the baud rate to use. If possible, set your serial port to 115200
baud to get good throughput.
GPS_SERIAL = <serialport>
<serialport>
value differs depending on how you have your gps receiver connected (by usb or gpio serial) and by SBC (RPi vs Nano)ls /dev/tty*
. Note that most of these are actually not usable./dev/ttyUSB0
./dev/ttyACM01
./dev/ttyAMA0
./dev/ttyTHS1
.GPS_SERIAL_BAUDRATE = <baudrate>
<baudrate>
value differs depending on your gps and if you have changed it using U-Center. Note that both the RPi and Jetson Nano may be using the default gpio serial port as a login console (you can connect up a serial 'terminal' and login). If using the gpio serial ports you need to disable the login console. See Writing to a serial port for details.
+Those two settings are the only ones related to the GPS receiver that need to be set in myconfig.py. Most GPS Receivers can also be directly configured to change things like the baudrate of the serial ports or how fast position estimates are sent to the computer. Ideally the rate of position estimates should be as fast as possible, but different receivers have different upper limits and there is some tradeoff between the rate of updates and how accurate they are. U-Blox based GPS receivers can be configured with U-Blox U-Center software; see the U-Center section of Donkeycar Meets RTK GPS for some details. Other chipset manufacturers have their own software; you will have to check your GPS receiver to determine the manufacturer. If you are using RTK high resolution GPS then you need to do a lot more configuration and wiring outside of Donkeycar. See Donkeycar meets RTK GPS for a detailed discussion of one way to setup an RTK GPS receiver for use with Donkeycar. Here is a related video that goes over the same information.
+An encoder setup can be used to estimate not only the vehicles' speed, but its position; that then allows encoders to be used with the Path Follow template in place of GPS, so it can be used indoors. This requires a few configurations to be set in the myconfig.py
; basically measurements of the wheel diameter, the length of the wheel base and the length of the axle. See Odometer Software Setup for details.
You can use either the web controller or a game controller. You can assign a game pad button OR web ui button to an action by editing the button assignments in myconfig.py. The name of the game pad buttons depend on the game controller you have configured (NOTE: one button is reserved for the emergency stop; you can see which one is assigned by looking at the console output when you start that car using the python manage.py drive
command). The 5 available web ui buttons are named web/w1
to web/w5
. If you assign None
action to a button then it is ignored.
SAVE_PATH_BTN
is the button to save the in-memory path to a file.LOAD_PATH_BTN
is the button to (re)load path from the csv file into memory.RESET_ORIGIN_BTN
is the button to set the current position as the origin.ERASE_PATH_BTN
is the button to erase path from memory and reset the origin.TOGGLE_RECORDING_BTN
is the button to toggle recording mode on or off. Note that there is a pre-assigned button in the web ui, so there is not need to assign this button to one of the web/w*
buttons if you are using the web ui.INC_PID_D_BTN
is the button to change PID 'D' constant by PID_D_DELTA. DEC_PID_D_BTN
is the button to change PID 'D' constant by -PID_D_DELTAINC_PID_P_BTN
is the button to change PID 'P' constant by PID_P_DELTADEC_PID_P_BTN
is the button to change PID 'P' constant by -PID_P_DELTAThe algorithm assumes we will be driving in a continuous connected path such that the start and end are the same. You can adjust the space between recorded waypoints by editing the PATH_MIN_DIST
value in myconfig.py You can change the name and location of the saved file by editing the PATH_FILENAME
value.
The workflow for recording a path is as follows:
+The path is saved as a comma-separated-values (.csv) file. Each line in the file contains 3 numbers separated by commas; x-position, y-position, throttle. The x and y positions are where the car was when the position was read and the throttle is the throttle value that was in effect at that time. Here is a section from a path file for illustration;
+0.0033510593930259347, 7.996719985734671, 0.14
+0.11206169077195227, 9.325505392625928, 0.16
+0.20344207028392702, 10.525161047000438, 0.18
+0.311049185693264, 11.724678185302764, 0.14
+0.23874327179510146, 12.75951695209369, 0.13
+0.26568955020047724, 14.015127370599657, 0.15
+0.35580877534812316, 15.06704786233604, 0.18
+0.4303318051388487, 16.192974457982928, 0.15
+0.2126157897291705, 17.302927474025637, 0.17
+-0.37973403913201764, 18.24986434960738, 0.17
+-1.2822835729457438, 18.97783037694171, 0.17
+-2.4313870034529828, 19.338536370545626, 0.17
+-3.633584696042817, 19.182584955357015, 0.17
+-4.694471199880354, 18.471380048431456, 0.25
+-5.2241318183369, 17.256997687276453, 0.25
+-5.462499356712215, 15.947787401732057, 0.25
+-5.5869644057238474, 14.674541235901415, 0.25
+
+Since the path is saved in a simple .csv file it can be visualized in many tools. A simple one to visualize your path is CSV Plot. Use the button in the upper-right (just to the left of the home button) to make the axis scale square. Here is an example path (rotated to fit a little better);
+ +The current autopilot uses a constant throttle value. You can set this by editing the PID_THROTTLE
value in myconfig.py.
The workflow for following a path is as follows:
+The algorithm we use for following the path is extremely simple; it's the Hello World of path following.
+PATH_SEARCH_LENGTH
points and choose the waypoint that is closest to the current position.PATH_LOOK_AHEAD
points ahead of the closest point on the path.PATH_LOOK_BEHIND
points behind the closes point on the path.In addition to steering, the path follow controller will set the throttle to the throttle saved with the closest point on the path scaled by the PID_THROTTLE
value in the myconfig.py
file. That can be overridden if USE_CONSTANT_THROTTLE = True
in the myconfig.py
, in which case it will use PID_THROTTLE
as the constant throttle.
So the algorithm uses the cross-track error between a desired line and the vehicle's measured position to decide how much and which way to steer. But the path we recorded is not a simple line; it is a lot of points that is typically some kind of circuit. As described above, we use the vehicle's current position to choose a short segment of the path that we use as our desired track. That short segment is recalculated every time we get a new measured car position. There are a few configuration parameters that determine exactly which two points on the path that we use to calculate the desired track line.
+PATH_SEARCH_LENGTH = None # number of points to search for closest point, None to search entire path
+PATH_LOOK_AHEAD = 1 # number of points ahead of the closest point to include in cte track
+PATH_LOOK_BEHIND = 1 # number of points behind the closest point to include in cte track
+
+Generally, if you are driving very fast you might want the look ahead to be larger than if driving slowly so that your steering can anticipate upcoming curves. Increasing the length of the resulting track line, by increasing the look behind and/or look ahead, also acts as a noise filter; it smooths out the track. This reduces the amount of jitter in the controller. However, this must be balanced with the true curves in the path; longer track segments effectively 'flatten' curves and so can result in understeer; not steering enough when on a curve.
+A PID controller is function that takes two parameters; 1) a target value to be achieved and 2) the current measured value. The PID function uses the difference between the target value and the measured value (the error) to calculate a control value that can be used to achieve the target value (so to drive the error between the desired value and the measured value to zero).
+In our case, we want to stay on the desired track; we want the cross-track error (the distance between the desired line and the vehicle's measured position) to be zero; the control value that is output is a steering value that should move the vehicle closer to the desired line. So our PID controller is controlling steering based on which side of the line and how far from the desired line the car is.
+The algorithm uses the sign of the cross track error to determine which way to steer. Naturally, if the cross-track error indicates the vehicle is to the left of the desired track, then the vehicle should turn right to move towards the desired track. If the cross-track error indicates the vehicle is to the right of the desired track, then the vehicle should turn left to move towards the desired track. If the vehicle is on the desired track, then the steering should be neutral.
+But how much should we steer; should we turn only slightly or should be turn very hard? The PID controller will output a steering value that is proportional to the magnitude of the cross-track error. So if we are near the desired track, then it will steer slightly. If we are far off the desired track then it will turn harder.
+The PID coefficients are the most important (and time consuming) parameters to configure. If they are not correct for your car then it will not follow the path. The coefficients can be changed by editing their values in the myconfig.py file.
+PID_P
is the proportional coefficient; it is multiplied with the cross-track error. This is the most important parameter; it contributes the most to the output steering value and in some cases may be all that is needed to follow the line. If this is too small then car will not turn enough when it reaches a curve. If this to too large then it will over-react to small changes in the path and may start turning in circles; especially when it gets to a curve.PID_D
is the differential coefficient; it is multiplied with the change in the cross-track error. This parameter can be useful in reducing oscillations and overshoot.PID_I
is the integral coefficient; it is multiplied with the total accumulated cross-track error. This may be useful in reducing offsets caused by accumulated error; such as if one wheel is slightly smaller in diameter than another.As descibed in the Configuring Button Actions section above, you can also assign functions like INC_PID_P_BTN
or DEC_PID_P_BTN
to the game controller or web ui buttons to modify the PID parameters 'on the fly'. This helps when you are figuring out the best coefficients. The button functions allow you to change values without having to stop the car, edit myconfig.py and restart the car.
Determining PID Coefficients can be difficult. One approach is:
+Be patient. Start with a reasonably slow speed. Change one thing at a time and test the change; don't make many changes at once. Write down what is working.
+Once you have a stable PID controller, then you can figure our just how fast you can go with it before autopilot becomes unstable. If you want to go faster then set the desired speed and start tweaking the values again using the method suggested above.
+Using the latest version of Raspian (tested with Raspian Buster) on the RPi, follow these instructions to set up Intel's Realsense libraries (Librealsense) and dependencies.
+Follow the standard instructions here
+After you’ve done that, set up the directory with this:
+```donkey createcar --path ~/follow --template path_follower
+Running
+cd ~/follow
+python3 manage.py drive
Once it’s running, open a browser on your laptop and enter this in the URL bar: http://
The rest of the instructions from Tawn’s repo:
+When you drive, this will draw a red line for the path, a green circle for the robot location.
+1) Mark a nice starting spot for your robot. Be sure to put it right back there each time you start. +2) Drive the car in some kind of loop. You see the red line show the path. +3) Hit X on the PS3/4 controller to save the path. +4) Put the bot back at the start spot. +5) Then hit the “select” button (on a PS3 controller) or “share” (on a PS4 controller) twice to go to pilot mode. This will start driving on the path. If you want it go faster or slower, change this line in the myconfig.py file: THROTTLE_FORWARD_PWM = 530
+Check the bottom of myconfig.py for some settings to tweak. PID values, map offsets and scale. things like that. You might want to start by downloading and using the myconfig.py file from my repo, which has some known-good settings and is otherwise a good place to start. +Some tips:
+When you start, the green dot will be in the top left corner of the box. You may prefer to have it in the center. If so, change PATH_OFFSET = (0, 0) in the myconfig.py file to PATH_OFFSET = (250, 250)
+For a small course, you may find that the path is too small to see well. In that case, change PATH_SCALE = 5.0 to PATH_SCALE = 10.0 (or more, if necessary)
+If you’re not seeing the red line, that means that a path file has already been written. Delete “donkey_path.pkl” (rm donkey_path.pkl) and the red line should show up
+When you're running in auto mode, the green dot will change to blue
+It defaults to recording a path point every 0.3 meters. If you want it to be smoother, you can change to a smaller number in myconfig.py with this line: PATH_MIN_DIST = 0.3
+ +Step 1: Setup Donkeycar
+Step 2: Setup Librealsense on Ubuntu Machine. +Using the latest version of Raspian (tested with Raspian Buster) on the RPi, follow these instructions to set up Intel's Realsense libraries (Librealsense) and dependencies.
+Step 3: Setup TensorRT on your Jetson Nano
+After you’ve done that, set up the directory with this:
+donkey createcar --path ~/follow --template path_follower
+
+Running
+cd ~/follow
+python3 manage.py drive
+
+Once it’s running, open a browser on your laptop and enter this in the URL bar: http://<your nano’s IP address>:8887
The rest of the instructions from Tawn’s repo:
+When you drive, this will draw a red line for the path, a green circle for the robot location.
+Mark a nice starting spot for your robot. Be sure to put it right back there each time you start.
+Drive the car in some kind of loop. You see the red line show the path.
+Hit X on the PS3/4 controller to save the path.
+Put the bot back at the start spot.
+Then hit the “select” button (on a PS3 controller) or “share” (on a PS4 controller) twice to go to pilot mode. This will start driving on the path. If you want it go faster or slower, change this line in the myconfig.py file: THROTTLE_FORWARD_PWM = 530
+Check the bottom of myconfig.py for some settings to tweak. PID values, map offsets and scale. things like that. You might want to start by downloading and using the myconfig.py file from my repo, which has some known-good settings and is otherwise a good place to start.
+Some tips:
When you start, the green dot will be in the top left corner of the box. You may prefer to have it in the center. If so, change PATH_OFFSET = (0, 0)
in the myconfig.py file to PATH_OFFSET = (250, 250)
For a small course, you may find that the path is too small to see well. In that case, change PATH_SCALE = 5.0
to PATH_SCALE = 10.0
(or more, if necessary)
If you’re not seeing the red line, that means that a path file has already been written. Delete “donkey_path.pkl” (rm donkey_path.pkl) and the red line should show up
+It defaults to recording a path point every 0.3 meters. If you want it to be smoother, you can change to a smaller number in myconfig.py with this line: PATH_MIN_DIST = 0.3
We have a different approaches to installing the software depending on the +version of Donkey Car. For Donkey Car <= 4.5.X we are using Jetpack 4.5.X +which comes with Tensorflow 2.3.1. The python installation is using virtual +env, i.e. it is based on the system python with version 3.6. This is the +only version that is working on the old Jetson Nano.
+For the main
branch we have updated Tensorflow to 2.9 and python to 3.8 or
+3.9. This is running on a newer version of Jetpack. You will need a Jetson
+Xavier or any of the newer Jetsons like the Orin, to use Jetpack 5.0.2. To
+decouple the python installation from the system python we are using Miniforge
+which is a mamba based version of Miniconda that works on the aarm architecture.
For Donkey Car <= 4.5.X please go to the next section. For the latest
+version on the main
branch please jump
+to this section.
We recommend to use 4GB version of the Jetson Nano or the Jetson Xavier to +run the software without issues. It's also recommended using a 128GB +microSD card with U3 speed, like for example +this SanDisk SD Card.
+These are the supported versions:
+Jetson | +Jetpack | +Python | +Donkey | +Tensorflow | +
---|---|---|---|---|
Nano | +4.5.1 | +3.6 | +<= 4.5.X | +2.3.1 | +
Xavier/Orin | +5.0.2 | +3.8 | +>= 5.X | +2.9 | +
Then Create your Donkeycar Application
+++Note: These instructions are working for DC 4.3.6 and DC 4.4.0 only at the +moment. We are working on a patch to have support for 4.5.X too.
+
These instructions work for Jetpack 4.5.1.
+This installs the official Nvidia build of Tensorflow 2.3.1; make sure you are +using the same version of Tensorflow on your host PC if you are using one. Using +a different version of Tensorflow to train your network may result in errors +when you attempt to use it as an autopilot.
+Visit the official Nvidia Jetson Nano Getting Started Guide +or Nvidia Xavier NX Getting Started Guide. +Work through the Prepare for Setup, Writing Image to the microSD Card, +and Setup and First Boot instructions, then return here.
+Once you're done with the setup, ssh into your vehicle. Use the terminal for +Ubuntu or Mac. Putty +for windows.
+Remove Libre Office:
+sudo apt-get remove --purge libreoffice*
+sudo apt-get clean
+sudo apt-get autoremove
+
+And add a 8GB swap file:
+git clone https://github.com/JetsonHacksNano/installSwapfile
+cd installSwapfile
+./installSwapfile.sh
+sudo reboot now
+
+sudo usermod -aG dialout <your username>
+sudo systemctl disable nvgetty
+
+First install some packages with apt-get
.
sudo apt-get update -y
+sudo apt-get upgrade -y
+sudo apt-get install -y libhdf5-serial-dev hdf5-tools libhdf5-dev zlib1g-dev zip libjpeg8-dev liblapack-dev libblas-dev gfortran
+sudo apt-get install -y python3-dev python3-pip
+sudo apt-get install -y libxslt1-dev libxml2-dev libffi-dev libcurl4-openssl-dev libssl-dev libpng-dev libopenblas-dev
+sudo apt-get install -y git nano
+sudo apt-get install -y openmpi-doc openmpi-bin libopenmpi-dev libopenblas-dev
+
+pip3 install virtualenv
+python3 -m virtualenv -p python3 env --system-site-packages
+echo "source ~/env/bin/activate" >> ~/.bashrc
+source ~/.bashrc
+
+Next, you will need to install packages with pip
:
pip3 install -U pip testresources setuptools
+pip3 install -U futures==3.1.1 protobuf==3.12.2 pybind11==2.5.0
+pip3 install -U cython==0.29.21 pyserial
+pip3 install -U future==0.18.2 mock==4.0.2 h5py==2.10.0 keras_preprocessing==1.1.2 keras_applications==1.0.8 gast==0.3.3
+pip3 install -U absl-py==0.9.0 py-cpuinfo==7.0.0 psutil==5.7.2 portpicker==1.3.1 six requests==2.24.0 astor==0.8.1 termcolor==1.1.0 wrapt==1.12.1 google-pasta==0.2.0
+pip3 install -U gdown
+
+# This will install tensorflow as a system package
+pip3 install --pre --extra-index-url https://developer.download.nvidia.com/compute/redist/jp/v45 tensorflow==2.5
+
+Change to a dir you would like to use as the head of your projects. Assuming
+you've already made the projects
directory above, you can use that. Get
+the latest 4.5.X release and install that into the venv.
mkdir projects
+cd ~/projects
+git clone https://github.com/autorope/donkeycar
+cd donkeycar
+git fetch --all --tags -f
+git checkout 4.5.1
+pip install -e .[nano]
+
+
+If you plan to use a USB camera, you will also want to setup pygame:
+sudo apt-get install python-dev libsdl1.2-dev libsdl-image1.2-dev libsdl-mixer1.2-dev libsdl-ttf2.0-dev libsdl1.2-dev libsmpeg-dev python-numpy subversion libportmidi-dev ffmpeg libswscale-dev libavformat-dev libavcodec-dev libfreetype6-dev
+pip install pygame
+
+Later on you can add the CAMERA_TYPE="WEBCAM"
in myconfig.py.
Instructions for the latest code from the main
branch or newer releases >=
+5.X. Note the installation differs between the two available OSs. On Jetson
+you need to install Jetpack 5.0.2.
These instructions work for Jetpack 5.0.2.
+Please install the Jetpack image from jetson-nx-developer-kit-sd-card-image.zip.
+Visit the official Nvidia Xavier NX Getting Started Guide. +Work through the Prepare for Setup, Writing Image to the microSD Card, +and Setup and First Boot instructions, then return here.
+Once you're done with the setup, ssh into your vehicle. Use the terminal for +Ubuntu or Mac. Putty +for windows.
+Remove Libre Office:
+sudo apt-get remove --purge libreoffice*
+sudo apt-get clean
+sudo apt-get autoremove
+
+And add a 8GB swap file. Note, if you intend to run from an SSD, perform the +swap file setup only after booting from the SSD:
+git clone https://github.com/JetsonHacksNano/installSwapfile
+cd installSwapfile
+./installSwapfile.sh -s 8
+reboot
+
+sudo usermod -aG dialout <your username>
+sudo systemctl disable nvgetty
+
+To install tensorflow and its dependencies for JP5.1.2 follow the NVIDIA instructions here
+python3 -m venv env --system-site-packages
+echo "source ~/env/bin/activate" >> ~/.bashrc
+source ~/.bashrc
+
+There are two different installations possible. Very likely you will +want to do the user install. Then you will perform Step 2a. In case you want +to debug or edit the source code, you will need to do the more advanced +developer install. But you can do only one.
+++Note: Only do Step 3b-4 or 3b-5 but not both!
+
As you have activated the new env
env already you type:
pip install donkeycar[nano]
+pip install -U albumentations --no-binary qudida,albumentations
+pip uninstall opencv-python-headless
+pip uninstall scikit-learn
+git clone https://github.com/scikit-learn/scikit-learn.git
+cd scikit-learn/
+python setup.py install
+sudo chmod 666 /dev/gpiochip*
+
+This will install the latest release.
+Only do this if you have not done the user install in 3b-4.
+Here you can choose which branch or tag you want to install, and you can
+edit and/or debug the code, by downloading the source code from GitHub. Do this
+for getting the latest version from the main
branch.
mkdir projects
+cd projects
+git clone https://github.com/autorope/donkeycar
+cd donkeycar
+git checkout main
+pip install -e .[nano]
+pip install -U albumentations --no-binary qudida,albumentations
+pip uninstall opencv-python-headless
+pip uninstall scikit-learn
+git clone https://github.com/scikit-learn/scikit-learn.git
+cd scikit-learn/
+python setup.py install
+sudo chmod 666 /dev/gpiochip*
+
+Run python and verify that tensorflow is version 2.9 and trt is version 8.2.1.
+To get the tensorrt shared libraries to load correctly we must set the
+environment variable LD_PRELOAD
as:
export LD_PRELOAD=/usr/lib/aarch64-linux-gnu/libnvinfer.so.8:/usr/lib/aarch64-linux-gnu/libgomp.so.1
+
+Note, this has to be done either every time you run donkeycar or tensorflow, or
+you put the above line into your .bashrc
.
python
+>>> import tensorflow as tf
+>>> tf.__version__
+>>> from tensorflow.python.compiler.tensorrt import trt_convert as trt
+>>> trt._check_trt_version_compatibility()
+>>> import cv2
+>>> print(cv2.getBuildInformation())
+
+If you plan to use a USB camera, you will also want to setup pygame:
+pip install pygame
+
+Later on you can add the CAMERA_TYPE="WEBCAM"
in myconfig.py.
This applies to any installation you did above, either JP 4.6.X or 5.0.X. +If you're using a CSIC camera you may have a pink tint on the images. As +described here, +this fix will remove it.
+wget https://www.dropbox.com/s/u80hr1o8n9hqeaj/camera_overrides.isp
+sudo cp camera_overrides.isp /var/nvidia/nvcam/settings/
+sudo chmod 664 /var/nvidia/nvcam/settings/camera_overrides.isp
+sudo chown root:root /var/nvidia/nvcam/settings/camera_overrides.isp
+
+Please read this carefully as Donkey Car is now installed differently +depending on the version. The latest Donkey Car version is 5.x and requires +64-bit Raspberry Pi OS Bullseye.
+If you're using an older version of Donkey Car, like 4.X then you need to +use the older Raspberry Pi OS (Raspian) version Buster, jump to +those instructions here.
+Tub data, car templates, etc are compatible between the two versions as well
+as models in keras format .h5
. However, Tensorflow Lite models .tflite
+are not and need to be regenerated.
In general, we recommend the RPi 4 with 4GB of ram. It's also recommended +using a 128GB microSD card with U3 speed, like for example +this SanDisk SD Card.
+This installation is using Raspberry Pi OS Bullseye (64 bit).
+Raspberry Pi OS can now be installed with the graphical installer Raspberry Pi +Imager which can be downloaded from here. +Please download and start the application, with the SD card you'll be using for your RaspberryPi inserted into your computer's SD card reader.
+First choose the device you'll be using: Raspberry Pi 5 or Raspberry Pi 4
+Then click on 'Operating System' and select 'Raspberry Pi OS (64 bit)'
+Then click on 'Storage' and select your SD card.
+Press 'NEXT' and you will be given the option to apply 'OS customization settings'. Select 'Edit Settings'
+Here you can enter the specifics of your username. password and wifi details. Set a hostname (here chosen to be +'donkeypi'), desired password, your wifi, region, etc. It should look like this: +
+Everything else can be left at the default. When you're done, click on 'Save' which will bring you back to the OS customization dialog. Click on 'Yes' and it will write the OS to your SD card.
+When it's done, you can place your SD card in the Pi and power it on. It will take a minute or so to boot the first time, but once it has done so (the green light stops flashing) You should be able to ssh into +the Pi through your network using the hostname 'donkeypi.local' (or whatever +you chose in the menu).
+sudo apt-get update --allow-releaseinfo-change
+sudo apt-get upgrade
+
+Doing it in the shell through:
+sudo raspi-config
+
+Interfacing Options
- I2C
Advanced Options
- Expand Filesystem
so you can use your whole
+ sd-card storageChoose <Finish>
and hit enter.
++Note: Reboot after changing these settings. Should happen if you select
+yes
.
Alternatively if you connect to the full desktop using VNC and are running +the desktop, go to 'Raspberry -> Preferences -> Raspberry Pi Configuration' +and apply the settings there.
+++Note: If you prefer to install the headless version of Raspberry Pi OS, +please follow the steps here. +You will need to run
+sudo apt -y install pip git
afterwards.
To create a virtual environmnet run the following from your home directory:
+python3 -m venv env --system-site-packages
+echo "source ~/env/bin/activate" >> ~/.bashrc
+source ~/.bashrc
+
+Install required libraries
+sudo apt install libcap-dev
+
+Create a project directory you would like to use as the
+head of your projects, change into it and download and install donkeycar
+from GitHub. Make sure your donkey
env is activated.
mkdir projects
+cd projects
+git clone https://github.com/autorope/donkeycar
+cd donkeycar
+git checkout main
+pip install -e .[pi]
+
+You can validate your tensorflow install with
+python -c "import tensorflow; print(tensorflow.__version__)"
+
+
+This installation is using Raspberry Pi OS Buster (32 bit).
+++Note: If you plan to use the mobile app, consider using the pre-built image. +Refer to the mobile app user guide for +details.
+
You need to flash a micro SD image with an operating system.
+++Note: Raspbian Latest (bullseye) is not compatible with the Python camera +bindings. The underlying camera system has changed. Please follow steps +below for installing the latest version from the
+main
branch.
We can create a special file which will be used to login to wifi on first boot. +More +reading here, +but we will walk you through it.
+On Windows, with your memory card image burned and memory disc still inserted, +you should see two drives, which are actually two partitions on the mem disc. +One is labeled boot. On Mac and Linux, you should also have access to the _ +_boot__ partition of the mem disc. +This is formatted with the common FAT type and is where we will edit some files +to help it find and log-on to your wifi on its first boot.
+++Note: If boot is not visible right away, try unplugging and re-inserting +the memory card reader.
+
gedit
on Linux.vi /Volumes/boot/wpa_supplicant.conf
where boot
is the
+ name of the SD Card).country
codes to use can be
+ found herecountry=US
+ctrl_interface=DIR=/var/run/wpa_supplicant GROUP=netdev
+update_config=1
+
+network={
+ ssid="<your network name>"
+ psk="<your password>"
+}
+
+
+Note - country
defines allowed wifi channels, ensure to set it properly to
+your location and hardware.
Replace <your network name>
with the ID of your network. Leave the quotes.
+I've seen problems when the network name contained an apostrophe, like "Joe's
+iPhone".
+Replace <your password>
with your password, leaving it surrounded by quotes.
+If it bothers you to leave your password unencrypted, you may change
+the contents later
+once you've gotten the pi to boot and log-in.
wpa_supplicant.conf
. On first boot, this file will be moved
+ to /etc/wpa_supplicant/wpa_supplicant.conf
where it may be edited later. If
+ you are using Notepad on Windows, make sure it doesn't have a .txt at the end.++Note: This step only possible on a Linux host pc. Otherwise you can set it up +later in
+raspi-config
after logging in to your pi.
We can also setup the hostname so that your Pi easier to find once on the +network. If yours is the only Pi on the network, then you can find it with
+ping raspberrypi.local
+
+once it's booted. If there are many other Pi's on the network, then this will
+have problems.
+If you are on a Linux machine, or are able to edit the UUID partition, then you
+can edit the /etc/hostname
and /etc/hosts
files now to make finding your pi
+on the network easier after boot.
+Edit those to replace raspberrypi
with a name of your choosing.
+Use all lower case, no special characters, no hyphens, yes underscores _
.
+Good idea is to use something like pi-<MAC_ADDRESS>
such as pi-deadbeef
+especially if you have more pi devices in the same network.
sudo vi /media/userID/UUID/etc/hostname
+sudo vi /media/userID/UUID/etc/hosts
+
+Put a file named ssh in the root of your boot partition. On Mac or Linux
+this can be done using the touch
command. For example, on the
+Mac, touch /Volumes/boot/ssh
where boot
is the name of the SD card.
Now your SD card is ready. Eject it from your computer - wait until system shows +the writing is done +and it is safe to remove card. Ensure Pi is turned off, put the card in the Pi +and power on the Pi.
+If you followed the above instructions to add wifi access, your Pi should +now be connected to your wifi network. Now you need to find its IP address +so you can connect to it via SSH.
+The easiest way (on Ubuntu) is to use the findcar
donkey command.
+You can try ping raspberrypi.local
. If you've modified the hostname, then you
+should try:
ping <your hostname>.local
+
+This will fail on a windows machine. Windows users will need the full IP +address (unless using cygwin).
+If you are having troubles locating your Pi on the network, you will want to +plug in an HDMI monitor and USB keyboard into the Pi. Boot it. Login with:
+pi
raspberry
Then try the command:
+ifconfig wlan0
+
+or just all Ip addresses assigned to the pi (wifi or cable):
+ip -br a
+
+If this has a valid IPv4 address, 4 groups of numbers separated by dots, then +you can try that with your SSH command. If you don't see anything like that, +then your wifi config might have a mistake. You can try to fix with
+sudo nano /etc/wpa_supplicant/wpa_supplicant.conf
+
+If you don't have a HDMI monitor and keyboard, you can plug-in the Pi with a +CAT5 cable to a router with DHCP. +If that router is on the same network as your PC, you can try:
+ping raspberrypi.local
+
+Hopefully, one of those methods worked and you are now ready to SSH into your +Pi. On Mac and Linux, you can open Terminal. +On Windows you can +install Putty, one of the alternatives, +or on Windows 10 you may have ssh via the command prompt.
+If you have a command prompt, you can try:
+ssh pi@raspberrypi.local
+
+or
+ssh pi@<your pi ip address>
+
+or via Putty.
+pi
raspberry
<your pi IP address>
sudo apt-get update --allow-releaseinfo-change
+sudo apt-get upgrade
+
+sudo raspi-config
+
+Interfacing Options
- I2C
Interfacing Options
- Camera
Advanced Options
- Expand Filesystem
so you can use your whole
+ sd-card storageChoose <Finish>
and hit enter.
++Note: Reboot after changing these settings. Should happen if you select
+yes
.
sudo apt-get install build-essential python3 python3-dev python3-pip python3-virtualenv python3-numpy python3-picamera python3-pandas python3-rpi.gpio i2c-tools avahi-utils joystick libopenjp2-7-dev libtiff5-dev gfortran libatlas-base-dev libopenblas-dev libhdf5-serial-dev libgeos-dev git ntp
+
+If you are going for a minimal install, you can get by without these. But it can +be handy to have OpenCV.
+sudo apt-get install libilmbase-dev libopenexr-dev libgstreamer1.0-dev libjasper-dev libwebp-dev libatlas-base-dev libavcodec-dev libavformat-dev libswscale-dev
+
+This needs to be done only once:
+python3 -m virtualenv -p python3 env --system-site-packages
+echo "source ~/env/bin/activate" >> ~/.bashrc
+source ~/.bashrc
+
+Modifying your .bashrc
in this way will automatically enable this environment
+each time you login. To return to the system python you can type deactivate
.
mkdir projects
+cd projects
+
+git clone https://github.com/autorope/donkeycar
+cd donkeycar
+git fetch --all --tags -f
+latestTag=$(git describe --tags `git rev-list --tags --max-count=1`)
+git checkout $latestTag
+pip install -e .[pi]
+pip install https://github.com/lhelontra/tensorflow-on-arm/releases/download/v2.2.0/tensorflow-2.2.0-cp37-none-linux_armv7l.whl
+
+You can validate your tensorflow install with
+python -c "import tensorflow; print(tensorflow.__version__)"
+
+If you've opted to install the OpenCV dependencies earlier, you can install +Python OpenCV bindings now with command:
+sudo apt install python3-opencv
+
+If that failed, you can try pip:
+pip install opencv-python
+
+Then test to see if import succeeds.
+python -c "import cv2"
+
+And if no errors, you have OpenCV installed!
+There is a mobile application available on the iPhone and Android that provides +an alternative user experience. It can be installed manually or by downloading +an SD card image. Follow +these instructions +to install manually.
+++Note The server component currently supports RaspberryPi 4B only.
+
18.04
. If you are
+ using Windows refer to
+ these
+ instructions on how to setup your computer to use TensorRT.Follow the instructions here.
+Make sure you use the tar
file instructions unless you have previously
+installed CUDA using .deb
files.
nvcc
is on $PATH
. Add the following
+ lines to your ~/.bashrc
file.# Add this to your .bashrc file
+export CUDA_HOME=/usr/local/cuda
+# Adds the CUDA compiler to the PATH
+export PATH=$CUDA_HOME/bin:$PATH
+# Adds the libraries
+export LD_LIBRARY_PATH=$CUDA_HOME/lib64:$LD_LIBRARY_PATH
+
+.bashrc
.source ~/.bashrc
+nvcc --version
+
+You should see something like:
+nvcc: NVIDIA (R) Cuda compiler driver
+Copyright (c) 2005-2018 NVIDIA Corporation
+Built on ...
+Cuda compilation tools, release 10.0, Vxxxxx
+
+virtualenv
and install PyCUDA.# This takes a a while.`
+pip install pycuda
+
+PYTHONPATH
such that
+ your dist-packages
are included as part of your virtualenv
. Add this to
+ your .bashrc
. This needs to be done because the python bindings
+ to tensorrt
are available in dist-packages
and this folder is usually not
+ visible to your virtualenv. To make them visible we add it to PYTHONPATH
.export PYTHONPATH=/usr/lib/python3.6/dist-packages:$PYTHONPATH
+
+virtualenv
and importing tensorrt
.> import tensorrt as trt
+> # This import should succeed
+
+uff
)After you train the linear
model you end up with a file with a .h5
+extension.
# You end up with a Linear.h5 in the models folder
+python manage.py train --model=./models/Linear.h5 --tub=./data/tub_1_19-06-29,...
+
+# (optional) copy './models/Linear.h5' from your desktop computer to your Jetson Nano in your working dir (~mycar/models/)
+
+# Freeze model using freeze_model.py in donkeycar/scripts ; the frozen model is stored as protocol buffers.
+# This command also exports some metadata about the model which is saved in ./models/Linear.metadata
+python ~/projects/donkeycar/scripts/freeze_model.py --model=~/mycar/models/Linear.h5 --output=~/mycar/models/Linear.pb
+
+# Convert the frozen model to UFF. The command below creates a file ./models/Linear.uff
+cd /usr/lib/python3.6/dist-packages/uff/bin/
+python convert_to_uff.py ~/mycar/models/Linear.pb
+
+Now copy the converted uff
model and the metadata
to your Jetson Nano.
myconfig.py
pick the model type as tensorrt_linear
.DEFAULT_MODEL_TYPE = `tensorrt_linear`
+
+# After you scp your `uff` model to the Nano
+python manage.py drive --model=./models/Linear.uff
+
+
+ Donkey supports three kinds of autopilots; a deep-learning autopilot, a path follow autopilot and a computer vision autopilot.
+If you followed along with the Create Donkeycar App section, then you know that you choose which template to use when you create your mycar application folder using the createcar command.
+This section will talk about what the templates are for, then we can get onto training an autopilot.
+The deep learning autopilot uses a single forward facing camera and a convolutional neural network to implement an autopilot using a technique known as Behavioral Cloning (also known as Imitation Learning). The technique is called Behavioral Cloning because the goal is to create an autopilot that imitates that actions of a human. This is the first kind of autopilot that Donkeycar supported and what it is best known for. For driving a car the overall process looks like this;
+Because the deep learning autopilot depends on a camera image, lighting conditions are important. The deep learning template is great for an indoor track where lighting conditions and the details of the room can be controlled, but it can be more difficult to get working outside where lighting conditions are variable and things change in the environment.
+Aside: The Donkeycar approach to deep learning driving was inspired by an Nvidia research paper entitled End to End Learning for Self-Driving Cars.
+Train a deep learning autopilot
+The path follow template is an alternative to the deep learning template. Outside we have access to GPS; the path follow template allows you to record a path using a GPS receiver and then configure an autopilot that can follow that path. The overall process looks like this;
+There is a lot more detail on this in the next section.
+ +The computer vision autopilot uses traditional computer vision techniques, such as color space conversion and edge detection algorithms, to identify features in the camera image and turn those into steering and throttle values. This autopilot has an advantage over the other autopilots in that it does not require manual driving to gather data. Instead you will choose or write a computer vision algorithm and modify the algorithm parameters to suit the track. This autopilot is specifically designed to make it easy to write your own algorithm using the OpenCV library and the many Donkeycar-provided primitives.
+There is a lot more detail in the next section about the build-in algorithm and how to write your own algorithm.
+ + +Donkey is a Self Driving Car Platform for hobby remote control cars. Donkey Car is made up of several components: +* It is a high level self driving library written in Python. It was developed with a focus on enabling fast experimentation and easy contribution. +* It is an Open Source Hardware design that makes it easy for you to build your own car +* It is a simulator that enables you to use Donkey without hardware +* It is a community of enthusiasts, developers and data scientists that enjoy racing, coding and discussing the future of ML, Cars and who will win the next race.
+Enjoy
+Donkey is the standard car that most people build first. The parts cost about $250 to $300 and take 2 hours to assemble. Here are the main steps to build your own car:
+Donkeycar is designed to make adding new parts to your car easy. Here's an +example car application that captures images from the camera and saves them.
+import donkey as dk
+
+#initialize the vehicle
+V = dk.Vehicle()
+
+#add a camera part
+cam = dk.parts.PiCamera()
+V.add(cam, outputs=['image'], threaded=True)
+
+#add tub part to record images
+tub = dk.parts.Tub(path='~/mycar/data',
+ inputs=['image'],
+ types=['image_array'])
+V.add(tub, inputs=inputs)
+
+#start the vehicle's drive loop
+V.start(max_loop_count=100)
+
+The ultimate goal of this project is to build something useful. Donkey's were +one of the first domesticated pack animals, they're notoriously stubborn, and +they are kid safe. Until the car can navigate from one side of a city to the +other, we'll hold off naming it after some celestial being.
+ +Donkeycar is very simple; code is organized into parts that take inputs and return outputs. These parts are added to a vehicle. The vehicle loop, once started, runs the parts in order. The parts effectively communicate by reading and mutating the vehicle memory.
+A 'template' is a python file that contains code to construct a 'vehicle' and one or more 'parts'. A part is a Python class that wraps a functional component of a vehicle. The parts are added to the vehicle. The parts can take values from the vehicle's memory as inputs and can write values to the vehicle's memory as outputs. When the vehicle loop is started the parts are run in the order that they were added; getting their inputs from memory and outputing their results to memory. This continues in a loop until the vehicle is stopped, then all the parts are shutdown and the template exits.
+When you create your car application using the donkey createcar ...
command as described in the Create Donkeycar App section of the docs, what happens under the hood is that a few files are copied from the donkeycar/templates
folder into your my car folder. The two we need to talk about are manage.py and myconfig.py.
The files that are copied to the mycar folder are renamed versions of a pair of template files in the templates folder. The files are chosen based on the template name you passed in the --template
argument to the createcar
command; if you pass nothing then the default is --template = complete
. So donkey createcar --path=~/mycar
is the same as donkey createcar --path=~/mycar --template=complete
. In this case then the files that are renamed and copied to ~/mycar/manage.py
and ~/mycar/myconfig.py
are donkeycar/templates/complete.py
and donkeycar/templates/cfg_complete.py
respectively. If you create a path follow application by passing --template=path_follow
to createcar, then the files that are copied are donkeycar/templates/path_follow.py
and donkeycar/templates/cfg_path_follow.py
Now technically another copy of the donkeycar/template/cfg_xxxx.py
is copied to the mycar folder as config.py
; this contains the default configuration and should not be edited. The myconfig.py file is really a commented out version of config.py. To change your app's configuration (like to choose the kind of camera or drivetrain) uncomment the section you care about in myconfig.py and edit it.
The manage.py file is where that action really is; this is the code that runs your car. It is organized into a 'vehicle loop' that runs at the rate specified by the DRIVE_LOOP_HZ
value in your myconfig.py file; that is how often the vehicle loop's 'parts' will get updated. The donkeycar vehicle loop is a pipeline of what we call 'parts' that get and set state in a hashmap we call 'memory'.
The complete.py and path_follow.py templates are fairly complex because they are very configurable. However they are not in anyway special. You can create your own template to do what you want; or you don't have create or use a template at all; you can write your own manage.py
directly. Here is an example of a vehicle loop with a single part that will accept a number, multiply it by a random number and return the result. As the vehicle loop runs, the value will continue to get randomized.
import random
+
+# the randomizer part
+class RandPercent:
+ def run(self, x):
+ value = x * random.random()
+ print(f"{x} -> {value}")
+ return value
+
+# create the vehicle and it's internal memory
+V = dk.Vehicle()
+
+# initialize the value in the vehicle's memory and give it a name
+V.mem['var'] = 4
+
+# add the part to read and write to the same value.
+V.add(RandPercent(), inputs=['var'], outputs=['var'])
+
+# start the vehicle loop running; quit after 5 loops
+V.start(max_loops=5)
+
+A part is a Python class that wraps a functional component of a vehicle.
+These include:
+Tawn Kramer has created a video (actual two parts) that walks through how to make a part. Also, there is a video of a presentationat the Arm AIoT conference that shows how the OLED part was created.
+Each part is constructed and then added to the vehicle loop with its named inputs and named outputs and an optional run_condition
specified. The vehicle's parts are (for the most part) executed in the order that they are added to the vehicle loop. Each time the vehicle loop runs the part's inputs are read from vehicle memory and passed to the part's run()
method, the run()
method does it's work, and it's return values are assigned to the output values. If there is a run_condition
, then the part's run()
method is only called in the value of the run_condition
property is True; so if the run_condition
property is False then the part is 'turned off'.
run()
method of a part; they are declared when the part is added to the vehicle loop. So for the aiLauncher example, when we add the part we include the argument, inputs=['user/mode', 'pilot/throttle']
. Just before the run()
method is called, the vehicle loop looks up the input values and then passes them to the part's run()
method as arguments. So when the aiLauncher part's run()
method is called it will be passed two arguments; the first will be the value of the user/mode
property in vehicle memory and the second will be the value of the pilot/throttle
property. Note that the number of inputs declared when the part is added must match the number of arguments in the part's run()
method otherwise a runtime error results. run()
method of the part; they are declared when the part is added to the vehicle loop. After the part's run()
method is called, the return values are assigned to named output properties. So for the aiLauncher example, when we add the part we include the argument, outputs=['pilot/throttle']
. When the aiLauncher part finishes running, it will return a single value and that value will be assigned to the 'pilot/throttle'
property in vehicle memory. Note that the number of outputs declared when the part is added must match the number of returned values in the part's run()
method otherwise a runtime error results. run_condition
is a boolean memory value that can be used to decide if a part's run()
method is called. If the condition is True then the part's run(
) method is called, otherwise it is not called. This is a way to turn on and off a part. So for instance, if we only ever wanted aiLauncher to run when in autopilot mode, we would maintain a named memory value, let's say 'run_pilot'
, that was True when running in autopilot mode and False when running in user (manual) mode. Then we would pass run_condition='run_pilot'
to the V.add()
method when we added the aiLauncher part to the vehicle. The aiLaucher's run()
method would only be called if the named memory value 'run_pilot'
was True.So you can see that you can control how a part operates by changing the value of its input properties. One part can affect another parts by outputting values (and so changing them) that are used as inputs or run_conditions by other parts.
+Here is an example of adding a part; the AiLaunch part overides the throttle when then driving mode transitions from manual driving to autopilot; it is used to provide a high throttle for a short time at the very start of a race. In this case it does not have an explicit run_condition
argument, so it defaults to True.
aiLauncher = AiLaunch(cfg.AI_LAUNCH_DURATION, cfg.AI_LAUNCH_THROTTLE, cfg.AI_LAUNCH_KEEP_ENABLED)
+ V.add(aiLauncher,
+ inputs=['user/mode', 'pilot/throttle'],
+ outputs=['pilot/throttle'])
+
+To implement this 'launch' it needs to know the current driving mode and the current autopilot throttle value; those are its inputs. If it is not launching then it just passes the throttle value through without modifying it, but when it is launching it outputs a throttle valud equal to cfg.AI_LAUNCH_THROTTLE
. So the throttle is it's only output. The part's run()
method must take in these two inputs in the correct order and return the one output. You can see this in part's code;
import time
+
+class AiLaunch():
+ '''
+ This part will apply a large thrust on initial activation. This is to help
+ in racing to start fast and then the ai will take over quickly when it's
+ up to speed.
+ '''
+
+ def __init__(self, launch_duration=1.0, launch_throttle=1.0, keep_enabled=False):
+ self.active = False
+ self.enabled = False
+ self.timer_start = None
+ self.timer_duration = launch_duration
+ self.launch_throttle = launch_throttle
+ self.prev_mode = None
+ self.trigger_on_switch = keep_enabled
+
+ def enable_ai_launch(self):
+ self.enabled = True
+ print('AiLauncher is enabled.')
+
+ def run(self, mode, ai_throttle):
+ new_throttle = ai_throttle
+
+ if mode != self.prev_mode:
+ self.prev_mode = mode
+ if mode == "local" and self.trigger_on_switch:
+ self.enabled = True
+
+ if mode == "local" and self.enabled:
+ if not self.active:
+ self.active = True
+ self.timer_start = time.time()
+ else:
+ duration = time.time() - self.timer_start
+ if duration > self.timer_duration:
+ self.active = False
+ self.enabled = False
+ else:
+ self.active = False
+
+ if self.active:
+ print('AiLauncher is active!!!')
+ new_throttle = self.launch_throttle
+
+ return new_throttle
+
+It is common for configuration values to be passed as arguments to a part's constructor, as they are in this example. Also, if the part grabs some hardware resource, such as a camera or a serial port, it should also have a shutdown()
function that releases those resources properly when donkey is stopped.
So as we said, the part's run()
method is called each time the vehicle loop is run; the input values are read from vehicle memory and passed as arguments to the run()
method, which does it's work and then returns values that are then written to vehicle memory as outputs. Since parts are run in the order they are added (for the most part) then you can see that you need to add a part that provides an output ahead of any part that needs that value as an input.
I say parts run in the order they were added 'for the most part' because you can also specify that a part is to be run in it's own thread so it can operate at it's own rate. A threaded part has a run_threaded()
method rather than a run()
method; the inputs are arguments and the return values are outputs just like the run()
method. Also similar to the run() method, the run_threaded() method is called once each time the vehicle loop runs.
So if run_threaded()
is called each time through the vehicle loop, just like the run()
method, and it's inputs and outputs are organized just the like a non-threaded part, then what is the difference between a threaded part and a non-threaded part? One difference you can see below is that when you add a threaded part you pass threaded=True
. But the most important difference with a threaded part is that it must have a no-argument update()
method. When the threaded part is launched a thread is created and the part's update()
method is registered as the method that is executed on the thread. The update()
method will run separately from the vehicle loop and it will run as fast as python's scheduler will allow; generally much faster than the vehicle loop runs. The update()
method should not return until the part is told to shutdown()
; it should run a loop that does its work over and over; such as reading from a device like the TFMini part does. In a threaded part the run_threaded()
method is usually quite simple; it typically just sets class properties used by the update()
method and returns class properties that are maintained by the update()
method.
Here is an example of adding a threaded part to the vehicle loop. This part interfaces to a TF-Mini single-beam lidar via a serial port; it reports a distance. The part takes no input arguments and outputs just the distance value. Note that the argument inputs=[]
is not really necessary; that is the default for inputs so it can be left off.
if cfg.HAVE_TFMINI:
+ from donkeycar.parts.tfmini import TFMini
+ lidar = TFMini(port=cfg.TFMINI_SERIAL_PORT)
+ V.add(lidar, inputs=[], outputs=['lidar/dist'], threaded=True)
+
+Here is a listing of the TFMini part;
+class TFMini:
+ """
+ Class for TFMini and TFMini-Plus distance sensors.
+ See wiring and installation instructions at https://github.com/TFmini/TFmini-RaspberryPi
+
+ Returns distance in centimeters.
+ """
+
+ def __init__(self, port="/dev/serial0", baudrate=115200, poll_delay=0.01, init_delay=0.1):
+ self.ser = serial.Serial(port, baudrate)
+ self.poll_delay = poll_delay
+
+ self.dist = 0
+
+ if not self.ser.is_open:
+ self.ser.close() # in case it is still open, we do not want to open it twice
+ self.ser.open()
+
+ self.logger = logging.getLogger(__name__)
+
+ self.logger.info("Init TFMini")
+ time.sleep(init_delay)
+
+ def update(self):
+ while self.ser.is_open:
+ self.poll()
+ if self.poll_delay > 0:
+ time.sleep(self.poll_delay)
+
+ def poll(self):
+ try:
+ count = self.ser.in_waiting
+ if count > 8:
+ recv = self.ser.read(9)
+ self.ser.reset_input_buffer()
+
+ if recv[0] == 0x59 and recv[1] == 0x59:
+ dist = recv[2] + recv[3] * 256
+ strength = recv[4] + recv[5] * 256
+
+ if strength > 0:
+ self.dist = dist
+
+ self.ser.reset_input_buffer()
+
+ except Exception as e:
+ self.logger.error(e)
+
+
+ def run_threaded(self):
+ return self.dist
+
+ def run(self):
+ self.poll()
+ return self.dist
+
+ def shutdown(self):
+ self.ser.close()
+
+++NOTE: The TFMini part manages a serial port itself; it is recommend to use the SerialPort part to read line oriented data from a serial port instead of managing the port in your part. The SerialPort part can handle all the details of the serial port and outputing the resulting data; then your part only needs to take that data as input and use it.
+
In the TFMini part the update()
method runs a loop as long as the serial port remains open. The serial port is opened in the constructor and closed when the shutdown()
method is called. In a threaded part, the update()
method is almost like an infinite loop, running over and over as much as python will give it time to run. This is the section of code that can run much faster than the vehicle loop runs.
The reason to use a threaded part is if your part needs to go faster than the vehicle loop or needs to respond to a device in close to real time. The loop in the update()
method will run as fast as the python interpreter can let it, which will usually be much faster than the vehicle loop. It's important to understand that the update()
method is called by the part's thread BUT the run_threaded()
method is called by the main vehicle loop thread. This means that these two methods may interupt each other in the middle of what they are doing.
You should use approprate thread-safe patterns, such as locks, to make sure that data updates and/or reads and other critical sections of code are safely isolated and atomic. In some cases this requires a Lock to make sure resources are accessed safely from threads or that multiple lines of code are executated atomically. It is worth remembering that assignment in Python is atomic (so there is one good thing about that Global Interpreter Lock, GIL). So while this is NOT atomic;
+x = 12.34
+y = 34.56
+angle = 1.34
+
+because your code could be interrupted in between those assignments. This IS atomic;
+pose = (12.34, 34.56, 1.34)
+
+So if you have aggregate internal state that may be mutated in a thread, then put it in a tuple and you can read and write it atomically without locks.
+ +A car needs a way to to move forward and backward and to turn left and right. We commonly call devices that produce a physical movement in the robot 'actuators'. Common actuators are DC motors, Servo motors, continuous servo motors and stepper motors. There are many, many different ways that these actuators can be combined to propel and turn a robot. Donkeycar supports two common configurations that can be implemented with various actuators:
+Actuators take control signals from the Donkeycar to control their actions. There are several options for generating these control signals.
+Below we will describe the supported actuator setups and software configuration of their control signals.
+A standard RC car is equipped with a steering servo for steering the front wheels and an ESC (Electronic Speed Controller) to control the speed of the DC motor driving the wheels. Both the steering servo and the ESC take a PWM (Pulse Width Modulation) control signal. A PWM signal is simply a square wave pulse of a certain duration and frequency. In the case of the steering servo the PWM signal determines the position of the servo's arm, which is generally between 0 degrees (full right) and 180 degrees (full left). In the case of the ESC the PWM signal determines the direction and speed of the drive motor, from full reverse, through stopped, to full forward.
+Configuration
+DRIVE_TRAIN_TYPE = "PWM_STEERING_THROTTLE"
in myconfig.pyPWM_STEERING_PIN
and PWM_THROTTLE_PIN
in the PWM_STEERING_THROTTLE
section of myconfig.py. For example: DRIVE_TRAIN_TYPE = "PWM_STEERING_THROTTLE"
+
+ #
+ # PWM_STEERING_THROTTLE
+ #
+ # Drive train for RC car with a steering servo and ESC.
+ # Uses a PwmPin for steering (servo) and a second PwmPin for throttle (ESC)
+ # Base PWM Frequence is presumed to be 60hz; use PWM_xxxx_SCALE to adjust pulse with for non-standard PWM frequencies
+ #
+ PWM_STEERING_THROTTLE = {
+ "PWM_STEERING_PIN": "PCA9685.1:40.0", # PCA9685, I2C bus 1, address 0x40, channel 0
+ "PWM_STEERING_SCALE": 1.0, # used to compensate for PWM frequency differents from 60hz; NOT for adjusting steering range
+ "PWM_STEERING_INVERTED": False, # True if hardware requires an inverted PWM pulse
+ "PWM_THROTTLE_PIN": "PCA9685.1:40.1", # PCA9685, I2C bus 1, address 0x40, channel 1
+ "PWM_THROTTLE_SCALE": 1.0, # used to compensate for PWM frequence differences from 60hz; NOT for increasing/limiting speed
+ "PWM_THROTTLE_INVERTED": False, # True if hardware requires an inverted PWM pulse
+ "STEERING_LEFT_PWM": 460, # pwm value for full left steering
+ "STEERING_RIGHT_PWM": 290, # pwm value for full right steering
+ "THROTTLE_FORWARD_PWM": 500, # pwm value for max forward throttle
+ "THROTTLE_STOPPED_PWM": 370, # pwm value for no movement
+ "THROTTLE_REVERSE_PWM": 220, # pwm value for max reverse throttle
+ }
+
+++NOTE: the pwm values (
+STEERING_LEFT_PWM
, etc.) differ from car to car and are derived by running the calibration procedure. See Calibrate your CarSee pins for a detailed discussion of pin providers and pin specifiers.
+
Configuration
+DRIVE_TRAIN_TYPE = "PWM_STEERING_THROTTLE"
in myconfig.py# PWM_STEERING_THROTTLE
section. Note that each pin has both a BOARD mode and a BCM (Broadcom) mode identifier. You can use either mode, but all pins must use the same mode.DRIVE_TRAIN_TYPE = "PWM_STEERING_THROTTLE"
and PWM_STEERING_PIN
and PWM_THROTTLE_PIN
are set to use one of the hardware pwm pins for output. For example: DRIVE_TRAIN_TYPE = "PWM_STEERING_THROTTLE"
+
+ #
+ # PWM_STEERING_THROTTLE
+ #
+ # Drive train for RC car with a steering servo and ESC.
+ # Uses a PwmPin for steering (servo) and a second PwmPin for throttle (ESC)
+ # Base PWM Frequence is presumed to be 60hz; use PWM_xxxx_SCALE to adjust pulse with for non-standard PWM frequencies
+ #
+ PWM_STEERING_THROTTLE = {
+ "PWM_STEERING_PIN": "RPI_GPIO.BOARD.33",# GPIO board mode pin-33 == BCM mode pin-13
+ "PWM_STEERING_SCALE": 1.0, # used to compensate for PWM frequency differents from 60hz; NOT for adjusting steering range
+ "PWM_STEERING_INVERTED": False, # True if hardware requires an inverted PWM pulse
+ "PWM_THROTTLE_PIN": "RPI_GPIO.BOARD.12",# GPIO board mode pin-12 == BCM mode pin-18
+ "PWM_THROTTLE_SCALE": 1.0, # used to compensate for PWM frequence differences from 60hz; NOT for increasing/limiting speed
+ "PWM_THROTTLE_INVERTED": False, # True if hardware requires an inverted PWM pulse
+ "STEERING_LEFT_PWM": 460, # pwm value for full left steering
+ "STEERING_RIGHT_PWM": 290, # pwm value for full right steering
+ "THROTTLE_FORWARD_PWM": 500, # pwm value for max forward throttle
+ "THROTTLE_STOPPED_PWM": 370, # pwm value for no movement
+ "THROTTLE_REVERSE_PWM": 220, # pwm value for max reverse throttle
+ }
+
+++NOTE: the pwm values (
+STEERING_LEFT_PWM
, etc.) differ from car to car and are derived by running the calibration procedure. See Calibrate your CarSee pins for a detailed discussion of pin providers and pin specifiers.
+
Please follow the instructions here
+Please follow the instructions here
+Arduino can be used in the following fashion to generate PWM signals to control the steering and throttle.
+For now the Arduino mode is only tested on the Latte Panda Delta (LP-D) board. +However it should be straightforward to use it with Raspberry Pi / Jetson Nano (instead of PCA 9685).
+Refer to the below block diagram to understand where things fits in.
+ +Arduino board should be running the standard firmata sketch (This sketch comes by default when you download the arduino tool). Load the standard firmata sketch (from Examples > Firmata > StandardFirmata) onto the Arduino. + +Further pymata_aio_ python package needs to be installed on the car computer via pip3 install pymata_aio.
+As shown in the block-diagram above LattePanda combines both the x86 CPU and the Connected Arduino into a single board.
+The following diagram shows how to connect the Arduino pins to steering servo and ESC.
++Note that the power for the servo is provided by the ESC battery elemininator circuit (BEC) which most ESC's provide. +This is done to avoid supplying the entire servo power from Arduino's 5v. +In large RC cars the servo can drag up to 2 amps, which lead to a destruction of the Arduino.
+Note that the calibration procedure/values are slightly different for the Arduino (than PCA9685). +Note that 90 is the usual midpoint (i.e. 1.5 ms pulse width at 50 Hz), so it is recommended to start + with 90 and adjust +/- 5 until you figure the desired range for steering / throttle.
+(env1) jithu@jithu-lp:~/master/pred_mt/lp/001/donkey$ donkey calibrate --arduino --channel 6
+using donkey v2.6.0t ...
+
+pymata_aio Version 2.33 Copyright (c) 2015-2018 Alan Yorinks All rights reserved.
+
+Using COM Port:/dev/ttyACM0
+
+Initializing Arduino - Please wait...
+Arduino Firmware ID: 2.5 StandardFirmata.ino
+Auto-discovery complete. Found 30 Digital Pins and 12 Analog Pins
+
+
+Enter a PWM setting to test(0-180)95
+Enter a PWM setting to test(0-180)90
+Enter a PWM setting to test(0-180)85
+...
+
+Note the --arduino switch passed to the calibrate command. Further note that the arduino pin being + calibrated is passed via the --channel parameter.
+The following snippet illustrates how to exercise the Arduino actuator in the drive() loop:
+ #Drive train setup
+ arduino_controller = ArduinoFirmata(
+ servo_pin=cfg.STEERING_ARDUINO_PIN,
+ esc_pin=cfg.THROTTLE_ARDUINO_PIN)
+
+ steering = ArdPWMSteering(controller=arduino_controller,
+ left_pulse=cfg.STEERING_ARDUINO_LEFT_PWM,
+ right_pulse=cfg.STEERING_ARDUINO_RIGHT_PWM)
+
+ throttle = ArdPWMThrottle(controller=arduino_controller,
+ max_pulse=cfg.THROTTLE_ARDUINO_FORWARD_PWM,
+ zero_pulse=cfg.THROTTLE_ARDUINO_STOPPED_PWM,
+ min_pulse=cfg.THROTTLE_ARDUINO_REVERSE_PWM)
+
+ V.add(steering, inputs=['user/angle'])
+ V.add(throttle, inputs=['user/throttle'])
+
+Refer to templates/arduino_drive.py for more details.
+In this configuration the DC motor that drives the wheels is controlled by an L298N HBridge motor controller or compatible. Steering the front wheels is accomplished with a Steering Servo that takes an PWM pulse. The motor driver is wired in one of two ways; 3 pin wiring or 2 pin wiring.
+A single DC gear motor is controlled with an L298N using two TTL output pins to select direction and a PWM pin to control the power to the motor.
+See https://www.electronicshub.org/raspberry-pi-l298n-interface-tutorial-control-dc-motor-l298n-raspberry-pi/ for a discussion of how the L298N HBridge module is wired in 3-pin mode to the RaspberryPi GPIO. This also applies to the some other driver chips that emulate the L298N, such as the TB6612FNG motor driver.
+Configuration
+DRIVETRAIN_TYPE = "SERVO_HBRIDGE_3PIN"
in myconfig.pyHBRIDGE_3PIN_FWD = "RPI_GPIO.BOARD.18" # ttl pin, high enables motor forward
+HBRIDGE_3PIN_BWD = "RPI_GPIO.BOARD.16" # ttl pin, highenables motor reverse
+HBRIDGE_3PIN_DUTY = "RPI_GPIO.BOARD.35" # provides duty cycle to motor
+PWM_STEERING_PIN = "RPI_GPIO.BOARD.33" # provides servo pulse to steering servo
+STEERING_LEFT_PWM = 460 # pwm value for full left steering (use `donkey calibrate` to measure value for your car)
+STEERING_RIGHT_PWM = 290 # pwm value for full right steering (use `donkey calibrate` to measure value for your car)
+
+A PCA9685 could also be used to generate all control signals. See pins for a detailed discussion of pin providers and pin specifiers.
+A single DC gear motor is controlled with an 'mini' L298N HBridge (or an L9110S HBridge) using 2 PWM pins; one pwm pin to enable and control forward speed and one to enable and control reverse motor speed.
+See https://www.instructables.com/Tutorial-for-Dual-Channel-DC-Motor-Driver-Board-PW/ for how an L298N mini-hbridge module is wired in 2-pin mode.
+See https://electropeak.com/learn/interfacing-l9110s-dual-channel-h-bridge-motor-driver-module-with-arduino/ for how an L9110S/HG7881 motor driver module is wired.
Configuration
+DRIVETRAIN_TYPE = "SERVO_HBRIDGE_2PIN"
in myconfig.py HBRIDGE_2PIN_DUTY_FWD = "RPI_GPIO.BOARD.18" # provides forward duty cycle to motor
+ HBRIDGE_2PIN_DUTY_BWD = "RPI_GPIO.BOARD.16" # provides reverse duty cycle to motor
+ PWM_STEERING_PIN = "RPI_GPIO.BOARD.33" # provides servo pulse to steering servo
+ STEERING_LEFT_PWM = 460 # pwm value for full left steering (use `donkey calibrate` to measure value for your car)
+ STEERING_RIGHT_PWM = 290 # pwm value for full right steering (use `donkey calibrate` to measure value for your car)
+
+A PCA9685 could also be used to generate all control signals. See pins for a detailed discussion of pin providers and pin specifiers.
+Some very inexpensive toy cars use a DC motor to drive the back wheels forward and reverse and another DC motor to steer the front wheels left or right. A single L298N HBridge (or L9110S HBridge) can be used to control these two motors. This driver assumes 2-pin wiring where each motor uses two PWM pins, one for each direction.
+Configuration
+DRIVETRAIN_TYPE = "DC_STEER_THROTTLE"
in myconfig.py HBRIDGE_PIN_LEFT = "RPI_GPIO.BOARD.18" # pwm pin produces duty cycle for steering left
+ HBRIDGE_PIN_RIGHT = "RPI_GPIO.BOARD.16" # pwm pin produces duty cycle for steering right
+ HBRIDGE_PIN_FWD = "RPI_GPIO.BOARD.15" # pwm pin produces duty cycle for forward drive
+ HBRIDGE_PIN_BWD = "RPI_GPIO.BOARD.13" # pwm pin produces duty cycle for reverse drive
+
+A PCA9685 could also be used to generate all control signals. See pins for a detailed discussion of pin providers and pin specifiers.
+VESC is an advanced version of ESC that provides you a lot of customization options on how the ESC operates. It includes features such as regenerative braking, temperature control etc.
+This was tested with a VESC 6 and Traxxas Brushless Motor +Follow this F1Tenth tutorial to update your VESC firmware and calibrate it: https://f1tenth.readthedocs.io/en/stable/getting_started/firmware/firmware_vesc.html +It's important to use the servo out bin so that we can control steering with the VESC as well
+Requires installation of PyVESC from source for servo control (pip install git+https://github.com/LiamBindle/PyVESC.git@master) +Configuration
+DRIVETRAIN_TYPE = "VESC"
in myconfig.py VESC_MAX_SPEED_PERCENT =.2 # Max speed as a percent of the actual speed
+ VESC_SERIAL_PORT= "/dev/ttyACM0" # Serial device to use for communication. Can check with ls /dev/tty*
+ VESC_HAS_SENSOR= True # Whether or not the bldc motor is using a hall effect sensor
+ VESC_START_HEARTBEAT= True # Whether or not to automatically start the heartbeat thread that will keep commands alive.
+ VESC_BAUDRATE= 115200 # baudrate for the serial communication. Shouldn't need to change this.
+ VESC_TIMEOUT= 0.05 # timeout for the serial communication
+ VESC_STEERING_SCALE= 0.5 # VESC accepts steering inputs from 0 to 1. Joystick is usually -1 to 1. This changes it to -0.5 to 0.5
+ VESC_STEERING_OFFSET = 0.5 # VESC accepts steering inputs from 0 to 1. Coupled with above change we move Joystick to 0 to 1
+
+An inexpensive Donkeycar compatible robot can be constructed using a cheap smart car robot chassis that includes 2 DC gear motors and an L298N motor driver or compatible to run the motors. Steering is accomplished by running one motor faster than the other, causing the car to drive in an arc. The motor driver can be wired in one of two ways; 3 pin wiring or 2 pin wiring. The name of the DonkeyCar drivetrains for differential drive all start with DC_TWO_WHEEL
.
2 DC gear motors are controlled with an L298N, each motor using two TTL output pins to select direction and a PWM pin to control the power to the motor. Since each motor uses 3 pins, so a total of 6 pins are used in a differential drive configuration. The advantage of this wiring scheme is that it only requires 2 PWM pins, which happens to be the maximum number of PWM pins on the Jetson Nano.
+See https://www.electronicshub.org/raspberry-pi-l298n-interface-tutorial-control-dc-motor-l298n-raspberry-pi/ for a discussion of how the L298N HBridge module is wired in 3-pin mode to the RaspberryPi GPIO. This also applies to the some other driver chips that emulate the L298N, such as the TB6612FNG motor driver.
+Configuration
+DRIVETRAIN_TYPE = "DC_TWO_WHEEL_L298N"
in myconfig.pyDC_TWO_WHEEL_L298N = {
+ "LEFT_FWD_PIN": "RPI_GPIO.BOARD.16", # TTL output pin enables left wheel forward
+ "LEFT_BWD_PIN": "RPI_GPIO.BOARD.18", # TTL output pin enables left wheel reverse
+ "LEFT_EN_DUTY_PIN": "RPI_GPIO.BOARD.22", # PWM pin generates duty cycle for left motor speed
+
+ "RIGHT_FWD_PIN": "RPI_GPIO.BOARD.15", # TTL output pin enables right wheel forward
+ "RIGHT_BWD_PIN": "RPI_GPIO.BOARD.13", # TTL output pin enables right wheel reverse
+ "RIGHT_EN_DUTY_PIN": "RPI_GPIO.BOARD.11", # PWM pin generates duty cycle for right wheel speed
+}
+
+DC_TWO_WHEEL_L298N = {
+ "LEFT_FWD_PIN": "PCA9685.1:40.3", # TTL output pin enables left wheel forward
+ "LEFT_BWD_PIN": "PCA9685.1:40.2", # TTL output pin enables left wheel reverse
+ "LEFT_EN_DUTY_PIN": "PCA9685.1:40.1", # PWM pin generates duty cycle for left motor speed
+
+ "RIGHT_FWD_PIN": "PCA9685.1:40.6", # TTL output pin enables right wheel forward
+ "RIGHT_BWD_PIN": "PCA9685.1:40.5", # TTL output pin enables right wheel reverse
+ "RIGHT_EN_DUTY_PIN": "PCA9685.1:40.4", # PWM pin generates duty cycle for right wheel speed
+}
+
+/opt/nvidia/jetson-io/jetson-io.py
. See Generating PWM from the Jetson Nano.++See pins for a detailed discussion of pin providers and pin specifiers.
+
2 DC Motors controlled with an 'mini' L293D HBridge, each motor using 2 PWM pins; one pwm pin to enable and control forward speed and one to enable and control reverse motor speed. This advantage of this wiring method is that it only requires a total of 4 pins; however all of those pins must be able to output PWM.
+Configuration
+DRIVETRAIN_TYPE = "DC_TWO_WHEEL"
in myconfig.pyDC_TWO_WHEEL = {
+ "LEFT_FWD_DUTY_PIN": "RPI_GPIO.BCM.16", # BCM.16 == BOARD.36, pwm pin produces duty cycle for left wheel forward
+ "LEFT_BWD_DUTY_PIN": "RPI_GPIO.BCM.20", # BCM.20 == BOARD.38, pwm pin produces duty cycle for left wheel reverse
+ "RIGHT_FWD_DUTY_PIN": "RPI_GPIO.BCM.5", # BCM.5 == BOARD.29, pwm pin produces duty cycle for right wheel forward
+ "RIGHT_BWD_DUTY_PIN": "RPI_GPIO.BCM.6", # BCM.6 == BOARD.31, pwm pin produces duty cycle for right wheel reverse
+}
+
+DC_TWO_WHEEL = {
+ "LEFT_FWD_DUTY_PIN": "PCA9685.1:40.0", # pwm pin produces duty cycle for left wheel forward
+ "LEFT_BWD_DUTY_PIN": "PCA9685.1:40.1", # pwm pin produces duty cycle for left wheel reverse
+ "RIGHT_FWD_DUTY_PIN": "PCA9685.1:40.5", # pwm pin produces duty cycle for right wheel forward
+ "RIGHT_BWD_DUTY_PIN": "PCA9685.1:40.6", # pwm pin produces duty cycle for right wheel reverse
+}
+
+++ +See pins for a detailed discussion of pin providers and pin specifiers.
+
Donkeycar supports a large number of cameras via the CAMERA_TYPE
configuration. For most applications a wide field of vision is important, so your camera should use a 120 degree wide angle lens or better. A 160 degree wide angle lense is recommended.
If you are using the default deep learning template or the computer vision template then you will need a camera. By default myconfig.py assumes a RaspberryPi camera. You can change this by editing the CAMERA_TYPE
value in the myconfig.py file in your ~/mycar
folder.
If you are using the gps path follow template then you do not need, and may not want, a camera. In this case you can change the camera type to mock; CAMERA_TYPE = "MOCK"
.
If you are on a raspberry pi and using the recommended pi camera ("PICAM"), then no changes are needed to your myconfg.py.
+This works with all Raspberry Pi cameras, including the original Raspberry Pi Camera Module based on the 5 megapixel OV5647 chipset and the Raspberry Pi Camera Module v2 based on the Sony IMX219 chip. These cameras are easily obtainable and are offered in generic (clone) versions by many vendors.
+The Jetson does not have a driver for the original 5 megapixels OV5647 based Raspberry Pi Camera, but it does have a driver for the v2 camera based on the IMX219 chip. Indeed the recommended camera is based on the IMX219 chip.
+The default setting CAMERA_TYPE = "PICAM"
does not work on the Jetson, even if you are using an 8mp RaspberryPi camera or a camera based on a Sony IMX219 based camera In either of these cases you will want edit your myconfg.py to have: CAMERA_TYPE = "CSIC"
.
For flipping the image vertically set CSIC_CAM_GSTREAMER_FLIP_PARM = 6
- this is helpful if you have to mount the camera in a rotated position.
CAMERA_TYPE = CVCAM
is a camera type that has worked for USB cameras when OpenCV is setup. This requires additional setup for OpenCV for Nano or OpenCV for Raspberry Pi.
If you have installed the optional pygame library then you can connect to the camera by editing the camera type to CAMERA_TYPE = "WEBCAM"
. See the required additional setup for pygame.
If you have more than one camera then you made need to the CAMERA_INDEX
configuration value. By default it is zero.
++++NOTE:
+CAMERA_TYPE = CVCAM
depends upon a version of OpenCV that has GStreamer support compiled in. This is the default on the Jetson computers and is supported in the recommended version of OpenCV for that Raspberry Pi.
The Intel Realsense cameras are RGBD cameras; they provide RGB images and Depth. You can use them as an RGB camera to provide images for the Deep Learning template or the Computer Vision template by setting CAMERA_TYPE = "D435"
in your myconfig.py settings. You will also want to review the settings that are specific to the Intel Realsense cameras;
# Intel Realsense D435 and D435i depth sensing camera
+REALSENSE_D435_RGB = True # True to capture RGB image
+REALSENSE_D435_DEPTH = True # True to capture depth as image array
+REALSENSE_D435_IMU = False # True to capture IMU data (D435i only)
+REALSENSE_D435_ID = None # serial number of camera or None if you only have one camera (it will autodetect)
+
+If you are not using depth then you will want to set REALSENSE_D435_DEPTH = False
so it does not save the depth data.
If the colors look wrong it may be that the camera is outputting BGR colors rather than RGB. You can set BGR2RGB = True
to convert from BGR to RGB.
We are adding other cameras over time, so read the camera section in myconfig.py to see what options are available.
+If you are having troubles with your camera, check out our Discord hardware channel for more help.
+ +The default controller to drive the car with your phone or browser. This has a web live preview of camera. Control options include:
+++Note: Recently iOS has disabled default Safari access to motion control.
+
If you bought an RC car then it might have come with a standard 2.4GHz car radio and receiver as shown in picture below. This can be used to drive the car. There are a few ways this can be accomplished.
+You can use a Teensy or Arduino microcontroller that can emulate a USB HID device and use this arduino sketch to make your RC controller emulate a game controller. Instructions for wiring and using the sketch can be found in the associated wiki. Once that is setup you would choose CONTROLLER_TYPE = "rc3"
as the controller type in your myconfig.py
configuration.
You can wire your RC car's receiver directly to the Raspberry Pi's gpio pins to read the length of the PWM steering and throttle signals being sent by your RC controller. Note that we don't recommend this for Jetson Nano users; the gpio support is not adequate. You can also use the RaspberryPi pins to output PWM directly to the car's servo and ESC, without the need for an I2C servo driver board. You will need to install the PiGPIO driver on you RaspberryPi to to make this work; it is not installed by default. If you wire your RC receiver to gpio pins, then you would choose CONTROLLER_TYPE = pigpio_rc
in your myconfig.py
configuration. A full tutorial on implementing the RC controller and servo/esc control via gpio pins is here.
Finally you can use the Donkeycar RC Hat. This board plugs into your RaspberryPi's gpio header and exposes connections for your RC Reciever's servo and throttle channels. It can also be used to control the car's steering servo and ESC, so you don't need a PCA9685 board. It includes a very handy OLED display that can be used to show your car's IP address on startup; see this DIYRobocars article. See the docs for how to setup the RC Hat. The Donkeycar RC Hat can be purchased in the Donkeycar Store.
+Many people find it easier to control the car using a game controller. There are several parts that provide this option.
+The default web controller may be replaced with a one line change to use a physical joystick part for input. This uses the OS device /dev/input/js0 by default. In theory, any joystick device that the OS mounts like this can be used. In practice, the behavior will change depending on the model of joystick ( Sony, or knockoff ), or XBox controller and the Bluetooth driver used to support it. The default code has been written and tested with a Sony brand PS3 Sixaxis controller. Other controllers may work, but will require alternative Bluetooth installs, and tweaks to the software for correct axis and buttons.
+These can be enabled by finding the CONTROLLER_TYPE in your myconfig.py and setting it to the correct string identifier ( after disabling the comment ).
+These can be used plugged in with a USB cable. It's been much more convenient to setup Bluetooth for a wireless control.
+There are controller specific setup details below.
+++Note: If you have a controller that is not listed below, or you are having troubles getting your controller to work or you want to map your controller differently, see Creating a New or Custom Game Controller.
+
python manage.py drive --js
+
+Will enable driving with the joystick. This disables the live preview of the camera and the web page features. If you modify myconfig.py to make USE_JOYSTICK_AS_DEFAULT = True
, then you do not need to run with the --js
.
Follow this guide. You can ignore steps past the 'Accessing the SixAxis from Python' section. I will include steps here in case the link becomes stale.
+sudo apt-get install bluetooth libbluetooth3 libusb-dev
+sudo systemctl enable bluetooth.service
+sudo usermod -G bluetooth -a pi
+
+Reboot after changing the user group.
+Plug in the PS3 with USB cable. Hit center PS logo button. Get and build the command line pairing tool. Run it:
+wget http://www.pabr.org/sixlinux/sixpair.c
+gcc -o sixpair sixpair.c -lusb
+sudo ./sixpair
+
+Use bluetoothctl to pair
+bluetoothctl
+agent on
+devices
+trust <MAC ADDRESS>
+default-agent
+quit
+
+Unplug USB cable. Hit center PS logo button.
+To test that the Bluetooth PS3 remote is working, verify that /dev/input/js0
exists:
ls /dev/input/js0
+
+In case the BT connection on the Raspberry Pi does not work, you see might something like this in bluetoothctl
:
[NEW] Controller 00:11:22:33:44:55 super-donkey [default]
+[NEW] Device AA:BB:CC:DD:EE:FF PLAYSTATION(R)3 Controller
+[CHG] Device AA:BB:CC:DD:EE:FF Connected: yes
+[CHG] Device AA:BB:CC:DD:EE:FF Connected: no
+[CHG] Device AA:BB:CC:DD:EE:FF Connected: yes
+[CHG] Device AA:BB:CC:DD:EE:FF Connected: no
+[CHG] Device AA:BB:CC:DD:EE:FF Connected: yes
+...
+[CHG] Device AA:BB:CC:DD:EE:FF Connected: yes
+[CHG] Device AA:BB:CC:DD:EE:FF Connected: no
+[bluetooth]#
+
+Try updating the Linux kernel and firmware by running:
+sudo rpi-update
+
+And then reboot:
+sudo reboot
+
+For some reason, they don't like to charge in a powered USB port that doesn't have an active Bluetooth control and OS driver. This means a phone type USB charger will not work. Try a powered Linux or mac laptop USB port. You should see the lights blink after plugging in and hitting center PS logo.
+After charging, you will need to plug-in the controller again to the Pi, hit the PS logo, then unplug to pair again.
+Sometimes these controllers can be quite old. Here's a link to a new battery. Be careful when taking off the cover. Remove 5 screws. There's a tab on the top half between the hand grips. You'll want to split/open it from the front and try pulling the bottom forward as you do, or you'll break the tab off as I did.
+Sometimes when you plug-in the PS3 joystick it starts taking over your mouse. If you want to prevent that, you can run this:
+xinput set-prop "Sony PLAYSTATION(R)3 Controller" "Device Enabled" 0
+
+The following instructions are intended for use with the Raspberry Pi 3 or 4 running Raspberry Pi OS Buster. +The DS4 gamepad will be connected via bluetooth without installing any additional software. Bluetoothd is a system service +that runs as a daemon automatically on boot. Bluetoothctl is a program to manage connection and pairing devices.
+Add the pi user to the bluetooth group. And then reboot so that the change takes effect properly.
+sudo usermod -a -G bluetooth pi
+sudo reboot
+
+After reboot, run bluetoothctl, turn on scanner to find bluetooth devices. See below for an example response. Note that +the actual HEX characters will be different for your devices!
+bluetoothctl
+<response> Agent registered
+<response> [bluetooth]#
+scan on
+<response>
+[CHG] Controller BB:22:EE:77:BB:CC Discovering: yes
+[NEW] Device 10:20:30:40:50:60 10-20-30-40-50-60
+[NEW] Device 10:20:30:40:50:70 10-20-30-40-50-70
+[NEW] Device 10:20:30:40:50:80 10-20-30-40-50-80
+[NEW] Device 10:20:30:40:50:90 10-20-30-40-50-90
+[NEW] Device 20:AA:88:44:BB:10 WHSCL1
+
+Wait a couple of minutes for the scanner to find all your existing bluetooth devices. Now set your gamepad in pairing mode +by holding down the share button and the playstation button together until the light double flashes. You should see a +new entry for a Wireless Controller.
+<response>
+[NEW] Device 1C:AA:BB:99:DD:AA Wireless Controller
+
+Turn off scanning to stop the status reporting
+scan off
+
+You will now connect, pair and trust the PS4 gamepad wireless controller. Trusting the paired devices will allow you to reconnect to the device after the Raspberry Pi reboots. Copy the wireless controller address. You will +type CONNECT "your wireless controller address", TRUST "your wireless controller address". In this case, "your wireless controller address" is 1C:AA:BB:99:DD:AA
+connect 1C:AA:BB:99:DD:AA
+<response>
+Attempting to connect to 1C:AA:BB:99:DD:AA
+[CHG] Device 1C:AA:BB:99:DD:AA Connected: yes
+[CHG] Device 1C:AA:BB:99:DD:AA UUIDs: 00001124-0000-1000-8000-00805f9b34fb
+[CHG] Device 1C:AA:BB:99:DD:AA UUIDs: 00001200-0000-1000-8000-00805f9b34fb
+[CHG] Device 1C:AA:BB:99:DD:AA ServicesResolved: yes
+[CHG] Device 1C:AA:BB:99:DD:AA Paired: yes
+Connection successful
+
+The PS4 gamepad light should now be solid. Now TRUST the PS4 gamepad wireless controller.
+trust 1C:AA:BB:99:DD:AA
+<response>
+[CHG] Device 1C:AA:BB:99:DD:AA Trusted: yes
+Changing 1C:AA:BB:99:DD:AA trust succeeded
+
+Type devices to see the paired-devices.
+paired-devices
+<response>
+Device 1C:AA:BB:99:DD:AA Wireless Controller
+
+Type quit or exit to quit the program bluetoothctl
+quit
+
+After booting your pi, press playstation button once. The light will flash for about 5 seconds and then turn solid. If +the light goes off, try again. If this does work, run bluetoothctl and verify devices and paired-devices.
+devices
+<response>
+Device 1C:AA:BB:99:DD:AA Wireless Controller
+
+paired-devices
+<response>
+Device 1C:A0:B8:9B:DB:A2 Wireless Controller
+
+If it fails to connect, while running bluetoothctl, press the playstation button once. A good response will be:
+<response>
+[CHG] Device 1C:AA:BB:99:DD:AA Connected: yes
+
+To disconnect the controller from the Raspberry Pi, press and hold the playstation button for 10 seconds.
+The following instructions are based on RetroPie and ds4drv.
+ds4drv
Running on your pi over ssh, you can directly install it:
+sudo /home/pi/env/bin/pip install ds4drv
+
+ds4drv
sudo wget https://raw.githubusercontent.com/chrippa/ds4drv/master/udev/50-ds4drv.rules -O /etc/udev/rules.d/50-ds4drv.rules
+sudo udevadm control --reload-rules
+sudo udevadm trigger
+
+ds4drv
ds4drv --hidraw --led 00ff00
+
+If you see Failed to create input device: "/dev/uinput" cannot be opened for writing
, reboot and retry. Probably granting permission step doesn't take effect until rebooting.
+Some controllers don't work with --hidraw
. If that's the case try the command without it. --led 00ff00
changes the light bar color, it's optional.
Press and hold Share button, then press and hold PS button until the light bar starts blinking. If it goes green after a few seconds, pairing is successful.
+ds4drv
in background on startup once bootedsudo nano /etc/rc.local
+
+paste:
+/home/pi/env/bin/ds4drv --led 00ff00
+
+Save and exit. Again, with or without --hidraw
, depending on the particular controller you are using.
To disconnect, kill the process ds4drv
and hold PS for 10 seconds to power off the controller.
This code presumes the built-in linux driver for 'Xbox Wireless Controller'; this is pre-installed on Raspbian, so there is no need to install any other drivers. This will generally show up on /dev/input/js0. There is another userland driver called xboxdrv; this code has not been tested with that driver.
+The XBox One controller requires that the bluetooth disable_ertm parameter be set to true; to do this:
+Adapted from: https://www.roboticsbuildlog.com/hardware/xbox-one-controller-with-nvidia-jetson-nano
+sudo apt-get install nano
+
+sudo usermod -a -G dialout $USER
+sudo reboot
+
+sudo apt-get install sysfsutils
+
+sudo nano /etc/sysfs.conf
+
+/module/bluetooth/parameters/disable_ertm=1
+
+sudo reboot
+
+Unpair (forget) the controller first if you already tried to pair it, then pair it again. You can do this with the Bluetooth Manager GUI appliation that ships with Jetpack or if you are using command line, then use bluetoothctl:
+Open terminal and type:
+ bluetoothctl
paired-devices
remove aa:bb:cc:dd:ee:ff
+ exit
sudo bluetoothctl
)Once paired you should have a solid light on the xbox button and a stable bluetooth connection.
+/etc/modprobe.d/xbox_bt.conf
(that may create the file; it is commonly not there by default)options bluetooth disable_ertm=1
disable_ertm
is set to true entering this
+ command in a terminal:bash
+ cat /sys/module/bluetooth/parameters/disable_ertm
Once that is done, you can pair your controller to your Raspberry Pi using the bluetooth tool. Enter the following command into a bash shell prompt:
+sudo bluetoothctl
+
+That will start blue tooth pairing in interactive mode. The remaining commands will be entered in that interactive session. Enter the following commands:
+agent on
+default-agent
+scan on
+
+That last command will start the Raspberry Pi scanning for new bluetooth devices. At this point, turn on your XBox One controller using the big round 'X' button on top, then start the pairing mode by pressing the 'sync' button on the front of the controller. Within a few minutes, you should see the controller show up in the output something like this;
+[NEW] Device B8:27:EB:A4:59:08 XBox One Wireless Controller
+
+Write down the MAC address, you will need it for the following steps. Enter this command to pair with your controller:
+connect YOUR_MAC_ADDRESS
+
+where YOUR_MAC_ADDRESS
is the MAC address you copied previously. If it does not connect on the first try, try again. It can take a few tries. If your controller connects, but then immediately disconnects, your disable_ertm setting might be wrong (see above).
Once your controller is connected, the big round 'X' button on the top of your controller should be solid white. Enter the following commands to finish:
+trust YOUR_MAC_ADDRESS
+quit
+
+Now that your controller is trusted, it should automatically connect with your Raspberry Pi when they are both turned on. If your controller fails to connect, run the bluetoothctl steps again to reconnect.
+To discover or modify the button and axis mappings for your controller, you can use the Joystick Wizard. The Joystick Wizard will write a custom controller named 'my_joystick.py' to your mycar folder. To use the custom controller, set CONTROLLER_TYPE="custom"
in your myconfig.py.
These parts encapsulate models defined using the FastAi high level api. They are intended to be used with the PyTorch backend. This allows you to build models using PyTorch or transfer learning.
+++Note This part is interchangeable with the Keras part but does not have TensorRT or TfLite support.
+
The parts are designed to use the trained artificial neural network to reproduce the steering and throttle given the image the camera sees. They are created by using the train command.
+This model type is created with the --type=fastai_linear
.
The FastAILinear
pilot uses one neuron to output a continuous value via the
+PyTorch Dense layer with linear activation. One each for steering and throttle.
+The output is not bounded.
Input: Image
+Network: 5 Convolution layers followed by two dense layers before output
+Output: Two dense layers with one scalar output each with linear activation for steering and throttle.
+ +IMUs or inertial measurement units are parts that sense the inertial forces on a robot. They vary depending on sensor, but may commonly include linear and rotational accelleration. They may sometimes include magnetometer to give global compasss facing dir. Frequently temperature is available from these as it affects their sensitivity.
+This is a cheap, small, and moderately precise imu. Commonly available at Amazon.
+MPU9250 offers additional integrated magnetometer.
+Install smbus
+ sudo apt install python3-smbus
+
+sudo apt-get install i2c-tools libi2c-dev python-dev python3-dev
+git clone https://github.com/pimoroni/py-smbus.git
+cd py-smbus/library
+python setup.py build
+sudo python setup.py install
+
+For MPU6050:
+Install pip lib for mpu6050
:
pip install mpu6050-raspberrypi
+
+For MPU9250:
+Install pip lib for mpu9250-jmdev
:
pip install mpu9250-jmdev
+
+Enable the following configurations to your myconfig.py
:
#IMU
+HAVE_IMU = True
+IMU_SENSOR = 'mpu9250' # (mpu6050|mpu9250)
+IMU_DLP_CONFIG = 3
+
+IMU_SENSOR
can be either mpu6050
or mpu9250
based on the sensor you are using.
IMU_DLP_CONFIG
allows to change the digital lowpass filter settings for your IMU. Lower frequency settings (see below) can filter high frequency noise at the expense of increased latency in IMU sensor data.
+Valid settings are from 0 to 6:
0
250Hz1
184Hz2
92Hz3
41Hz4
20Hz 5
10Hz6
5HzAt startup the MPU9250 driver performs calibration to zero accel and gyro bias. Usually the process takes less than 10 seconds, and in that time avoid moving or touching the car. +Please place the car on the ground before starting Donkey.
+ +These parts encapsulate models defined using the Keras high level api. They are intended to be used with the Tensorflow backend. The parts are designed to use the trained artificial neural network to reproduce the steering and throttle given the image the camera sees. They are created by using the train command.
+This model type is created with the --type=categorical
.
The KerasCategorical
pilot breaks the steering and throttle decisions into discreet angles and then uses categorical cross entropy to train the network to activate a single neuron for each steering and throttle choice. This can be interesting because we get the confidence value as a distribution over all choices.
+This uses the dk.utils.linear_bin
and dk.utils.linear_unbin
to transform continuous real numbers into a range of discreet values for training and runtime.
+The input and output are therefore bounded and must be chosen wisely to match the data.
+The default ranges work for the default setup. But cars which go faster may want to enable a higher throttle range. And cars with larger steering throw may want more bins.
This model was the original model, with some modifications, when Donkey was first created.
+Input: Image
+Network: 5 Convolution layers followed by two dense layers before output
+Output: Two dense layers, 16, and 20 w categorical output
+This model type is created with the --type=linear
.
The KerasLinear
pilot uses one neuron to output a continous value via the
+Keras Dense layer with linear activation. One each for steering and throttle.
+The output is not bounded.
Input: Image
+Network: 5 Convolution layers followed by two dense layers before output
+Output: Two dense layers with one scalar output each with linear activation for steering and throttle.
+This model type is created with the --type=imu
.
The KerasIMU
pilot is very similar to the KerasLinear
model, except that it takes intertial measurment data in addition to images when learning to drive.
+This gives our stateless model some additional information about the motion of the vehicle.
This can be a good starting point example of ingesting more data into your models.
+Input: Image, vector of linear and angular acceleration
+Network: 5 Convolution layers followed by two dense layers before output, Vector data is followed by 3 dense layers then concatenating before 2 dense control layers and after conv2d layers.
+Output: Two dense layers with one scalar output each with linear activation for steering and throttle.
+This model type is created with the --type=latent
.
The KerasLatent
pilot tries to force the model to learn a latent vector in addition to driving. This latent vector is a bottleneck in a CNN that then tries to reproduce the given input image and produce driving commands. These dual tasks could produce a model that learns to distill the driving scene and perhaps better abstract to a new track.
Input: Image
+Network: 5 Convolution layers bottleneck to a 10x1x1 vector, followed by 6Conv2dTranspose layers before outputing to a image and 3 dense layers and driving controls.
+Output: Two dense layers with one scalar output each with linear activation for steering and throttle. Outputs an image.
+This model type is created with the --type=rnn
.
The KerasRNN
pilot uses a sequence of images to control driving rather than just a single frame. The number of images used is controlled by the SEQUENCE_LENGTH
value in myconfig.py.
Input: Image
+Network: 4 time distributed Convolution layers, followed by 2 LSTM layers, 3 dense layers, and driving controls.
+Output: One dense layer with two scalar outputs for steering and throttle.
+This model type is created with the --type=3d
.
The Keras3D_CNN pilot uses a sequence of images to control driving rather than just a single frame. The number of images used is controlled by the SEQUENCE_LENGTH value in myconfig.py. Instead of 2d convolutions like most other models, this uses a 3D convolution across layers.
+Input: Image
+Network: 4 3D Convolution layers each followed by max pooling, followed by 2 dense layers, and driving controls.
+Output: One dense layer with two scalar outputs for steering and throttle.
+This model type is created with the --type=behavior
.
The KerasBehavioral pilot takes an image and a vector as input. The vector is one hot activated vector of commands. This vector might be of length two and have two states, one for left lane driving and one for right lane driving. Then during training one element of the vector is activated while the desired behavior is demonstrated. This vector is defined in myconfig.py BEHAVIOR_LIST
. BEHAVIOR_LED_COLORS
must match the same length and can be useful when showing the current state. TRAIN_BEHAVIORS
must be set to True.
Input: Image, Behavior vector
+Network: 5 Convolution layers, followed by 2 dense layers, and driving controls.
+Output: Categorical steering, throttle output similar to Categorical keras model.
+This model type is not created without some code modification.
+The KerasLocalizer pilot is very similar to the Keras Linear model, except that it learns to output it's location as a category. +This category is arbitrary, but has only been tested as a 0-9 range segment of the track. This requires that the driving data is marked up with a category label for location. This could supply some higher level logic with track location, for driving stategy, lap counting, or other.
+Input: Image
+Network: 5 Convolution layers followed by two dense layers before output
+Output: Two dense layers with one scalar output each with linear activation for steering and throttle. One categorical output for location.
+ +A Lidar sensor can be used with Donkeycar to provide obstacle avoidance or to help navigate on tracks with walls. It records data along with the camera during training and this can be used for training
+NOTE: Lidar is currently only supported in the Dev branch. To use it, after you git clone donkeycar, do a git checkout dev
+
We currently only support the RPLidar series of sensors, but will be adding support for the similar YDLidar series soon.
+We recommend the $99 A1M8 (12m range)
+Mount the Lidar underneath the camera canopy as shown above (the RPLidar A2M8 is used there, but the A1M8 mounting is the same). You can velcro the USB adapter under the Donkey plate and use a short USB cable to connect to one of your RPi or Nano USB ports. It can be powered by the USB port so there's no need for an additional power supply.
+Lidar requires the glob library to be installed. If you don't already have that, install it with pip3 install glob2
Also install the Lidar driver: pip install Adafruit_CircuitPython_RPLIDAR
Then go to the lidarcar directory and edit the myconfig.py file to ensure that the Lidar is turned on. The upper and lower limits should be set to reflect the areas you want your Lidar to "look at", omitting the areas that are blocked by parts of the car body. An example is shown below. For the RPLidar series, 0 degrees is in the direction of the motor (in the case of the A1M8) or cable (in the case of the A2M8)
+# LIDAR
+USE_LIDAR = True
+LIDAR_TYPE = 'RP' #(RP|YD)
+LIDAR_LOWER_LIMIT = 90 # angles that will be recorded. Use this to block out obstructed areas on your car and/or to avoid looking backwards. Note that for the RP A1M8 Lidar, "0" is in the direction of the motor
+LIDAR_UPPER_LIMIT = 270
+
++
Neither the deep learning template nor the path follow template supports Lidar data directly. There is an issue to add Lidar data to the deep learning template. Lidar would also be very useful in the path follow template for obstacle detection and avoidance. If you are interested in working on such projects, please join the discord community and let us know; we will be happy to provide you with support.
+ +Odometry is a way to calculate the speed and distance travelled of the car by measuring the rotation of its wheels using a sensor called an rotary encoder. This encoder can be on the motor, on the main drive shaft or on individual wheels. The advantage of using an encoder is that it "closes the loop" with your throttle, so your car can reliably command an actual velocity rather than just issuing a motor control which will produce a faster or slower velocity depending on the slope of the track, the surface or mechanical friction in your drive train while turning. In short, an encoder gives you much better control over your speed.
+Encoders come in various forms
+There are several ways to read encoders with Donkey:
+Arduino: +The recommended way is with an Arduino compatible microcontroller running one of the Arduino sketches included with Donkeycar. Since the microcontroller is dedicated to counting pulses it can maintain an accurate count event with very high resolution encoders. This is critical if you are using a high resolution encoder so it does not drop any encoder pulses and so undercount. There are two Arduino sketches available:
+Both sketches support a single encoder or two encoders in a differential drive arrangement. They can be compiled using interrupts for high resolution encoders or as polled-encoders with robust debouncing. Both sketches transmit the count to the RPi via the USB serial port when requested by Donkeycar, which lightens the processing load for the Rpi.
+GPIO: +If you are using a low-resolution mono-encoder attached to the output shaft of the motor or to the drive shaft then the Raspberry Pi's GPIO pins may be adequate to count the pulses. Remember that the GPIO pins only support 3.3v devices; if you can supply your encoder's VCC at 3.3v then it will generally output 3.3v pulses, in which case you can directly connect it to the RaspberryPi's GPIO pins.
+Examples of rotary encoders that are supported:
+How you attach your encoder is up to you and which kind of encoder you're using. For example, here's one way to put a quadrature encoder on the main drive shaft. Here is a more complex setup with dual encoders.
+But this is the easiest way to do it, with a cheap and simple optical encoder on the main drive shaft of a standard Donkeycar chassis (if your chassis is different, the same overall approach should work, although you may have to find a different place to mount the sensor):
+First, unscrew the plate over the main drive shaft. Tilt the rear wheels back a bit and you should be able to remove the shaft.
+ +Now enlarge the hole in the optical encoder disc that came with your sensor (use a drill or Dremel grinding stone) so you can slip it onto the shaft. Stretch a rubber grommet (you can use the sort typically included with servos to mount them, but any one of the right size will do) over the shaft and push it into the encoder disc hole. If you don't have a grommet, you can wrap tape around the shaft until it's large enough to hold the disc firmly. Once you've ensured it's in the right place, use a few drops of superglue or hot glue to hold it in place)
+ + +Cut out a small notch (marked in pencil here) in the plate covering the drive shaft, so you can mount the encoder sensor there, ensuring that the disc can turn freely in the gap in front of the steering servo
+ +Now replace the plate and drill two holes so you can screw in the encoder sensor. Slide the disc along the shaft so that it doesn't bind on the sensor.
+ +Use three female-to-female jumper cables and connect the sensor to your RPi GPIO pins as follows. Connect the GND, V+ (which might say 5V or 3.3V) and data pin (which will say "Out or "D0") to the RPi 5V, Ground and GPIO 13 as shown here (if your sensor encoder has four pins, ignore the one that says "A0"): +
+Note: if you're already using GPIO 13 for another reason, such as RC input or output, you can use any other free GPIO pin. Just change the ODOM_PIN
number accordingly in the myconfig.py
file as shown below.
Enable odometry in myconfig.py
.
#
+# ODOMETRY
+#
+HAVE_ODOM = False # Do you have an odometer/encoder
+HAVE_ODOM_2 = False # Do you have a second odometer/encoder as in a differential drive robot.
+ # In this case, the 'first' encoder is the left wheel encoder and
+ # the second encoder is the right wheel encoder.
+ENCODER_TYPE = 'GPIO' # What kind of encoder? GPIO|arduino.
+ # - 'GPIO' refers to direct connect of a single-channel encoder to an RPi/Jetson GPIO header pin.
+ # Set ODOM_PIN to the gpio pin, based on board numbering.
+ # - 'arduino' generically refers to any microcontroller connected over a serial port.
+ # Set ODOM_SERIAL to the serial port that connects the microcontroller.
+ # See 'arduino/encoder/encoder.ino' for an Arduino sketch that implements both a continuous and
+ # on demand protocol for sending readings from the microcontroller to the host.
+ENCODER_PPR = 20 # encoder's pulses (ticks) per revolution of encoder shaft.
+ENCODER_DEBOUNCE_NS = 0 # nanoseconds to wait before integrating subsequence encoder pulses.
+ # For encoders with noisy transitions, this can be used to reject extra interrupts caused by noise.
+ # If necessary, the exact value can be determined using an oscilliscope or logic analyzer or
+ # simply by experimenting with various values.
+FORWARD_ONLY = 1
+FORWARD_REVERSE = 2
+FORWARD_REVERSE_STOP = 3
+TACHOMETER_MODE=FORWARD_REVERSE # FORWARD_ONLY, FORWARD_REVERSE or FORWARD_REVERSE_STOP
+ # For dual channel quadrature encoders, 'FORWARD_ONLY' is always the correct mode.
+ # For single-channel encoders, the tachometer mode depends upon the application.
+ # - FORWARD_ONLY always increments ticks; effectively assuming the car is always moving forward
+ # and always has a positive throttle. This is best for racing on wide open circuits where
+ # the car is always under throttle and where we are not trying to model driving backwards or stopping.
+ # - FORWARD_REVERSE uses the throttle value to decide if the car is moving forward or reverse
+ # increments or decrements ticks accordingly. In the case of a zero throttle, ticks will be
+ # incremented or decremented based on the last non-zero throttle; effectively modelling 'coasting'.
+ # This can work well in situations where the car will be making progress even when the throttle
+ # drops to zero. For instance, in a race situatino where the car may coast to slow down but not
+ # actually stop.
+ # - FORWARD_REVERSE_STOP uses the throttle value to decide if the car is moving forward or reverse or stopped.
+ # This works well for a slower moving robot in situations where the robot is changing direction; for instance4
+ # when doing SLAM, the robot will explore the room slowly and may need to backup.
+MM_PER_TICK = WHEEL_RADIUS * 2 * 3.141592653589793 * 1000 / ENCODER_PPR # How much travel with a single encoder tick, in mm. Roll you car a meter and divide total ticks measured by 1,000
+ODOM_SERIAL = '/dev/ttyACM0' # serial port when ENCODER_TYPE is 'arduino'
+ODOM_SERIAL_BAUDRATE = 115200 # baud rate for serial port encoder
+ODOM_PIN = 13 # if using ENCODER_TYPE=GPIO, which GPIO board mode pin to use as input
+ODOM_PIN_2 = 14 # GPIO for second encoder in differential drivetrains
+ODOM_SMOOTHING = 1 # number of odometer readings to use when calculating velocity
+ODOM_DEBUG = False # Write out values on vel and distance as it runs
+
+If you are using an Arduino compatible microcontroller to read your encoder, set ENCODER_TYPE = 'arduino'
in the myconfig.py file. The microcontroller should be flashed using the Arduino IDE with one of the sketches in the arduino folder. The sketches can be checked in the Arduino IDE by using the serial console after flashing the microcontroller. The sketches implement the r/p/c command protocol for on-demand sending of encoder value and continuous sending with provided delay. Commands are sent one per line (ending in '\n'):
r
command resets position to zerop
command sends position immediatelyc
command starts/stops continuous modeWith a single encoder setup the encoder sends the tick count and a timestamp as a comma delimited pair over the serial/USB port:
+{ticks},{ticksMs}
In a dual encoder setup the second encoder values as separated from the first by a semicolon:
+{ticks},{ticksMs};{ticks},{ticksMs}
The tachometer.py
file that implements the encoder parts also has a __main__
function, so it can be run directly. After activating the donkey
python environment the file can be run to check your hookup and to determine configuration parameters. Run this to get the available arguments:
python donkeycar/parts/tachometer.py
+
+An encoder setup can be used to estimate not only the vehicle's speed, but its position. This requires a few configurations to be set in the myconfig.py
; basically measurements of the wheel diameter, the length of the wheel base and the length of the axle. This then allows encoders to be used with the Path Follow template in place of GPS, so it can be used indoors.
#
+# MEASURED ROBOT PROPERTIES
+#
+AXLE_LENGTH = 0.03 # length of axle; distance between left and right wheels in meters
+WHEEL_BASE = 0.1 # distance between front and back wheels in meters
+WHEEL_RADIUS = 0.0315 # radius of wheel in meters
+MIN_SPEED = 0.1 # minimum speed in meters per second; speed below which car stalls
+MAX_SPEED = 3.0 # maximum speed in meters per second; speed at maximum throttle (1.0)
+MIN_THROTTLE = 0.1 # throttle (0 to 1.0) that corresponds to MIN_SPEED, throttle below which car stalls
+MAX_STEERING_ANGLE = 3.141592653589793 / 4 # for car-like robot; maximum steering angle in radians (corresponding to tire angle at steering == -1)
+
+
+ OLED displays can be used to show information about the current state of the car. This is especially useful in the when collecting data for training, and when racing.
+The OLED display currently displays the following information:
+* The IP address of the car (eth
and wlan
)
+* The number of records collected, for training.
+* The driving mode.
Examples of displays that are currently supported are:
+ +Simply connect the display to the I2C pins on the Raspberry Pi or the Jetson Nano. Use bus 1
so the display can be inserted directly on the pins. Here is an example of what that looks like.
Enable the display in myconfig.py
by uncommenting this line USE_SSD1306_128_32 = False
by removing the #
at the start and change False
to True
. If you have a 128x32 OLED select resolution 1, if you have 128x64 select resolution 2 and don't forget to remove the #
in front of that line, too, to make it active.
This part of your myconfig.py
file should now look like this.
USE_SSD1306_128_32 = True # Enable the SSD_1306 OLED Display
+# SSD1306_128_32_I2C_ROTATION = 0 # 0 = text is right-side up, 1 = rotated 90 degrees clockwise, 2 = 180 degrees (flipped), 3 = 270 degrees
+SSD1306_RESOLUTION = 2 # 1 = 128x32; 2 = 128x64
+
+One of the cool things about having an OLED screen is that you can show your car's IP address on startup, so you can connect to it. Instructions to set that up are here
+If you are unable to start the car, ensure that the Adafruit_SSD1306
package is installed in your virtual environment. This should automatically be installed, if you are using a recent version of donkeycar
.
pip install Adafruit_SSD1306
+
+Adafruit_SSD1306
library is incompatible with steering/motor configurations when the Duty Cycle/PWM is supplied directly from GPIO header using the RPI_GPIO
pin provider. This is because internally the Adafruit library sets a GPIO pin mode that is incompatible with our GPIO library. In this case you have a couple of options:PIGPIO
pin provider to generate the necessary duty cycle/PWM for throttle and steering from the GPIO. See PIGPIO for how to set up the pigpio library.Rather than using a standard camera and training a network to drive, Donkeycar supports using the Intel Realsense T265 "tracking camera" to follow a path instead. In this application, you simply drive a path once manually, and Donkeycar will "remember" that path and repeat it autonomously.
+The Intel T265 uses a combination of stereo cameras and an internal Inertial Measurement Unit (IMU) plus its own Myriad X processor to do Visual Inertial Odometry, which is a fancy way of saying that it knows where it is by looking at the scene around it as it moves and correlating that with the IMU's sensing to localize itself, outputting an X,Y,Z position to Donkey, much as a GPS sensor would (but ideally much more accurately, to a precision of centemeters)
+Using the latest version of Raspian (tested with Raspian Buster) on the RPi, follow these instructions to set up Intel's Realsense libraries (Librealsense) and dependencies. Although those instructions discuss another Realsense sensor, they work equally well for the T265. There are also video instructions
+Follow the standard instructions here. With the Path Follower, there is no need to install Tensorflow for this particular Donkeycar configuration however do install numpy/upgrade before running "pip install -e .[pi]"
+```donkey createcar --path ~/follow --template path_follow
+cd ~follow
+sudo nano myconfig.py
Make sure you agree with the default values or adjust them to your liking (ie. "throttle", "steering", PIDs, etc.). Uncomment (remove the #) for any line you've changed. In Nano press cntrl-o to save the file and cntrl-x to exit.
+Running
+ssh pi@<your pi’s IP address or "raspberrypi.local">
+cd ~/follow
+python3 manage.py drive
Keep the terminal open to see the printed output of the app while it is running.
+If you get an error saying that it can't find the T265, unplug the sensor, plug it back in and try again. Ensure that your gamepad is on and connected, too (blue light is on the controller)
+Once it’s running, open a browser on your laptop and enter this in the URL bar: http://
When you drive, the Web interface will draw a red line for the path, a green circle for the robot location. If you're seeing the green dot but not the red line, that means that a path file has already been written. Delete “donkey_path.pkl” (rm donkey_path.pkl), restart and the red line should show up
+PS4 Gamepad controls are as follows: ++------------------+--------------------------+ +| control | action | ++------------------+--------------------------+ +| share | toggle auto/manual mode | +| circle | save_path | +| triangle | erase_path | +| cross | emergency_stop | +| L1 | increase_max_throttle | +| R1 | decrease_max_throttle | +| options | toggle_constant_throttle | +| square | reset_origin | +| L2 | dec_pid_d | +| R2 | inc_pid_d | +| left_stick_horz | set_steering | +| right_stick_vert | set_throttle | ++------------------+--------------------------+
+1) Mark a nice starting spot for your robot. Be sure to put it right back there each time you start.
+2) Drive the car in some kind of loop. You see the red line show the path.
+3) Hit circle on the PS3/4 controller to save the path.
+4) Put the bot back at the start spot.
+5) Then hit the “select” button (on a PS3 controller) or “share” (on a PS4 controller) twice to go to pilot mode. This will start driving on the path. If you want it go faster or slower, change this line in the myconfig.py file: THROTTLE_FORWARD_PWM = 400
Check the bottom of myconfig.py for some settings to tweak. PID values, map offsets and scale. things like that. You might want to start by downloading and using the myconfig.py file from my repo, which has some known-good settings and is otherwise a good place to start.
+Some tips:
+When you start, the green dot will be in the top left corner of the box. You may prefer to have it in the center. If so, change PATH_OFFSET = (0, 0) in the myconfig.py file to PATH_OFFSET = (250, 250)
+For a small course, you may find that the path is too small to see well. In that case, change PATH_SCALE = 5.0 to PATH_SCALE = 10.0 (or more, if necessary)
+When you're running in auto mode, the green dot will change to blue
+It defaults to recording a path point every 0.3 meters. If you want it to be smoother, you can change to a smaller number in myconfig.py with this line: PATH_MIN_DIST = 0.3
+ +Control signals are send and received by pins on the Raspberry Pi, Jetson Nano and connected peripherals, like the PCA9685 Servo controller. Starting wit version 5.x, Donkeycar uses 'pin specs' to specify pins including various configuration that is specific to the underlying hardware or library implementation. This allows use to make the underlying logic, like how a motor controller takes throttle values and outputs them to the motor, more independent of the particular hardware or library used to generate the signals.
+PWM pins generate a square wave, sometimes called a PWM pulse. This is used to control servo motors, electronic speed controllers, and LEDs.
+TTL Output pins can generate either a high value (1) or a low value (0)
+TTL Input pins read values as either high (1) or low (0)
+Donkeycar supports several technologies for specifying pins. Pins are specified as a string that identifies the provider, the pin number and any techology specific configuration.
+The PCA9685 Servo controller supports 16 PWM and TTL output pins. The PCA9685 can only output signals; it does not support input pins. The pin specifier for a PCA9685 pin includes:
+For example, "PCA9685.1:40.13"
specifies channel 13 on the PCA9685 on I2C bus 1 at address 0x40.
For example, "PCA9685.0:60.1"
specified channel 1 on the PCA9685 on I2C bus 0 at address 0x60
Donkeycar installs the RPi.GPIO library on the RaspberryPi in the default installation. The Jetson.GPIO library is compatible library installed by default on the Jetson Nano. Both of these libaries work is a similar fashion to support PWM, input and output pins on the 40 pin GPIO bus of the RaspberryPi or Jetson Nano respectively. The pin specifier includes:
+See details of the RaspberryPi 40 pin header here: https://www.raspberrypi.com/documentation/computers/os.html#gpio-and-the-40-pin-header
+Jetson Nano 40 pin header uses the same board numbering scheme, although the header is physically flipped on the board, so pay attention to the numbers printed on the board. The Jetson Nano only supports 2 PWM pins and these must be enabled. See Generating PWM from the Jetson Nano
+For example, "RPI_GPIO.BOARD.33"
specifies board pin 33 using the Rpi.GPIO library.
For example, "RPI_GPIO.BCM.13"
specifies Broadcom GPIO-13 using the Rpi.GPIO library. If you look at the header diagram linked above you will notice that this is the same physical pin as "RPI_GPIO.BOARD.33"; it is a synonymn for physical pin 33.
When using the RPI_GPIO pin provider, you can choose to use the BOARD or BCM pin schemes, but all pins must use the same pin scheme. You cannot mix pin schemes.
+RaspberryPi users can optionally install the PiGPIO library and daemon to manage the pins on the 40 pin GPIO header. Note that this library does NOT work on the Jetson Nano. The library support PWM, Input and Output pins.
+sudo apt-get update
+sudo apt-get install pigpio
+
+pip install pigpio
+
+sudo systemctl start pigpiod
+
+sudo systemctl enable pigpiod
+
+The PIGPIO pin specifier includes: +- "BCM" PiGPIO used Broadcom (BCM) pin numbering scheme exclusively, so that is baked into the pin specifier. +- The BCM pin number
+For example, "PIGPIO.BCM.13"
specifies Broadcom GPIO-13. As discussed above and shown in the linked header diagram, this is exposed on board pin 33.
Both the Jetson Nano and RaspberryPi4 support two hardware PWM pins. On the Jetson Nano, these must be configured.
+ssh into the donkeycar and run this command sudo /opt/nvidia/jetson-io/jetson-io.py
. It should show the Jetson Expansion Header Tool that allows you to change GPIO pin functions (see below).
If your Jetson expansion header configuration does not show any PWM pins, then you will need to enable them.
+---
+ =================== Jetson Expansion Header Tool ===================
+ | |
+ | |
+ | 3.3V ( 1) ( 2) 5V |
+ | i2c2 ( 3) ( 4) 5V |
+ | i2c2 ( 5) ( 6) GND |
+ | unused ( 7) ( 8) uartb |
+ | GND ( 9) (10) uartb |
+ | unused (11) (12) unused |
+ | unused (13) (14) GND |
+ | unused (15) (16) unused |
+ | 3.3V (17) (18) unused |
+ | unused (19) (20) GND |
+ | unused (21) (22) unused |
+ | unused (23) (24) unused |
+ | GND (25) (26) unused |
+ | i2c1 (27) (28) i2c1 |
+ | unused (29) (30) GND |
+ | unused (31) (32) unused |
+ | unused (33) (34) GND |
+ | unused (35) (36) unused |
+ | unused (37) (38) unused |
+ | GND (39) (40) unused |
+ | |
+ ====================================================================
+---
+
+Choose Configure the 40 pin expansion header
to activate pwm0 and pwm2:
---
+ =================== Jetson Expansion Header Tool ===================
+ | |
+ | |
+ | 3.3V ( 1) ( 2) 5V |
+ | i2c2 ( 3) ( 4) 5V |
+ | i2c2 ( 5) ( 6) GND |
+ | unused ( 7) ( 8) uartb |
+ | GND ( 9) (10) uartb |
+ | unused (11) (12) unused |
+ | unused (13) (14) GND |
+ | unused (15) (16) unused |
+ | 3.3V (17) (18) unused |
+ | unused (19) (20) GND |
+ | unused (21) (22) unused |
+ | unused (23) (24) unused |
+ | GND (25) (26) unused |
+ | i2c1 (27) (28) i2c1 |
+ | unused (29) (30) GND |
+ | unused (31) (32) pwm0 |
+ | pwm2 (33) (34) GND |
+ | unused (35) (36) unused |
+ | unused (37) (38) unused |
+ | GND (39) (40) unused |
+ | |
+ ====================================================================
+---
+
+After enabling, pwm0 is board pin-32 and pwm2 is board pin-33.
+ ++(This only works with the RaspberryPi. The Jetson Nano does not provide the necessary GPIO pin support)
+You can drive Donkey with nothing more than the RC controller your car probably came with! The secret is that, thanks to the cool Pigpio library, the RaspberryPi pins can read and generate the RC signals necessary to read your RC receiver and drive your servo and motor controllers.
+To do this you need to either connect some jumper cables from your RC receiver to the RPi GPIO pins and then do the same for your steering servo and motor controller (it's a little fiddly but works fine) or use our Donkeycar RC Hat (shown above), which is plug and play and includes other nice stuff like a OLED screen, a fan, encoder support and even an e-stop option (like a remote kill switch) if you happen to have a 3Ch (or more) RC transmitter.
+If you're using the RC Hat above, you can skip this hardware part -- the hat does it all for you!
+Note that you will want your RC controller to be well trimmed prior to using it to control your Donkeycar. You want the throttle trim, steering trim and steering range to be well adjusted; see this video for how to do that.
+You can use the GPIO pins for RC input, output or both. In the case of RC input, the RC controller replaces a bluetooth joystick. In the case of RC output, it replaces the I2C servo driver board.
+The easiest way to connect RC is via the custom "hat" that we've designed (see above). But if you're doing it yourself, follow this wiring guide. It's a bit of a forest of jumper cables if you're doing both input and output, but remember that you only have to connect one ground and V+ cable to the RC reciever (on any channel), rather than one for every channel.
+Also note the the RC receiver should be connected to the 3.3v pins, while the output servo and motor controller are connected to the 5v pins.
+Warning: The RC receiver PWM signal is generated from the receiver input voltage, so connecting the RC receiver to 5V or even 6V from the ESC will fry the RPi!
+ +Here's what the RC receiver connection should look like
+ +First, make sure PIGPIO is installed; see pins You probably want the PIGPIO daemon to allows be started whe the RaspberrpyPi start. On the command line enter this to set the PIGPIO daemon to always run on startup:
+sudo systemctl enable pigpiod & sudo systemctl start pigpiod
+
+Next, in your mycar
directory, edit the myconfig.py files as follows:
pigpio_rc
as your controller type in your myconfig.py file. Uncomment the line (remove the leading #
) and edit it as follows:CONTROLLER_TYPE = 'pigpio_rc'
+
+Also set use joystick
to True
USE_JOYSTICK_AS_DEFAULT = True
+
+PWM_STEERING_THROTTLE
as your drive train type in your myconfig.py file. Uncomment the line (remove the leading #
) and edit it as follows:DRIVE_TRAIN_TYPE = "PWM_STEERING_THROTTLE"
+
+For both of these, there are additional settings you can change, such as reversing the direction of output or the pins connected:
+Input options:
+#PIGPIO RC control
+STEERING_RC_GPIO = 26
+THROTTLE_RC_GPIO = 20
+DATA_WIPER_RC_GPIO = 19
+PIGPIO_STEERING_MID = 1500 # Adjust this value if your car cannot run in a straight line
+PIGPIO_MAX_FORWARD = 2000 # Max throttle to go fowrward. The bigger the faster
+PIGPIO_STOPPED_PWM = 1500
+PIGPIO_MAX_REVERSE = 1000 # Max throttle to go reverse. The smaller the faster
+PIGPIO_SHOW_STEERING_VALUE = False
+PIGPIO_INVERT = False
+PIGPIO_JITTER = 0.025 # threshold below which no signal is reported
+
+If you are using the RC hat then the PWM output pins shown below (and defaulted in myconfig.py) must be used. +If you are not using the RC hat then you are free to choose different PWM output pins. +NOTE: you must install pigpio to use this configuration. See PIGPIO
+Output options:
+PWM_STEERING_PIN = "PIGPIO.BCM.13" # PWM output pin for steering servo
+PWM_THROTTLE_PIN = "PIGPIO.BCM.18" # PWM output pin for ESC
+
+STEERING_LEFT_PWM = int(4096 * 1 / 20) # pwm value for full left steering (1ms pulse)
+STEERING_RIGHT_PWM = int(4096 * 2 / 20) # pwm value for full right steering (2ms pulse)
+
+THROTTLE_FORWARD_PWM = int(4096 * 2 / 20) # pwm value for max forward (2ms pulse)
+THROTTLE_STOPPED_PWM = int(4096 * 1.5 / 20) # pwm value for no movement (1.5ms pulse)
+THROTTLE_REVERSE_PWM = int(4096 * 1 / 20) # pwm value for max reverse throttle (1ms pulse)
+
+If one channel is reversed (steering left goes right, etc), either reverse that channel on your RC transmitter (that's usually a switch or setting) or change it in the output options shown above by channging the PWM_INVERTED value for that channel to True
.
If you started with a ready-to-run RC car, it probably came with a RC controller. Good news: you can use it with Donkeycar, using the RC controller for manual driving. You can also plug in the car's servo and motor controller directly into the RaspberryPi without the need for a PCA9685 motor/servo controller board.
+Note that you will want your RC controller to be well trimmed prior to using it with the RC hat. You want the throttle trim, steering trim and steering range to be well adjusted; see this video for how to do that.
+To do so, you can either wire up it up manually as shown in this tutorial (which works, but has a lot of fiddly wires that can fall off) or do it far more neatly with the Donkeycar RC hat, shown above, which handles all the wiring for you, along with including an OLED screen and a fan.
+The Donkeycar RC hat can be purchased from the Donkeycar Store. Note that it only works with the RaspberryPi, not the Jetson Nano, due to limitations with the way the Jetson handles its I/O pins.
+If you're using a standard wheel encoder, you can plug it into the "Encoder" pins. You can also power the RaspberryPi from this board if you have a 5V source with the "Optional 5v power in" pins
+Once you've plugged in all the cables, you can move to the software setup
+There are two parts to the software setup. The first part is setting up to read the RC Controller using the RC Hat. The Second, optional, part is setting up the drive train so we can control the ESC and SERVO using the RC hat (so you don't need a PCA9685 anymore)/
+In both cases we are going to use the PiGPIO library to control the I/O pins (remember, this only works on a RaspberryPi). Install PiGPIO from a command prompt as follows;
+sudo apt-get update
+sudo apt-get install pigpio
+
+Then, on the command line enter this to set the PIGPIO daemon to always run on startup:
+sudo systemctl enable pigpiod & sudo systemctl start pigpiod
+
+The RC Hat can route the PWM signals generated by your RC Receiver to the RaspberryPi's gpio pins, so software can measure the length of the PWM pulse and then use that to determine steering and throttle. This allows you to use the RC controller that came with your RC rather than using a game controller.
+Connection: +To use the RC hat to read your RC controller, use the included 3-wire cables to connect your RC receiver to the RC 1 and RC 2 pins (corresponding to the RC receiver's Channel 1 and Channel 2). In all cases, make sure you plug them in the right way, noting the +,- and S (Signal) markings. Typically the black wire is "-", the red wire in the middle is "+" and the white wire is "S".
+Configuration:
+Now edit your myconfig.py file to use the RC Hat to read the RC Controller. In your mycar
directory, edit the myconfig.py files as follows:
Use pigpio_rc
as your controller type in your myconfig.py file. Uncomment the CONTROLLER_TYPE
line (remove the leading #
) and edit it as follows:
CONTROLLER_TYPE = 'pigpio_rc'
+
+Also set use joystick
to True
USE_JOYSTICK_AS_DEFAULT = True
+
+There are additional settings you can change in the #PIGPIO RC control
section, such as reversing the direction of output or the pins connected, or adjusting the expect PWM pulse width (see Standard RC with ESC and Steering Servo for a discussion of Pulse Width Modulation); TLDR - a 1000 nanosecond pulse means full left/reverse, a 1500 nano second pulse means straight/stopped and a 2000 nanosecond pulse means full right/forward. The defaults are generally good and you can start with them. If you see any issues when calibrating then read the Troubleshooting
section to see how you might change one or more of these values to compensate.
Input options for reading RC controller:
+#PIGPIO RC control
+STEERING_RC_GPIO = 26 # gpio pin (in broadcom numbering) for reading the RC controller's steering
+THROTTLE_RC_GPIO = 20 # gpio pin (in broadcom numbering) for reading the RC Controller's throttle
+DATA_WIPER_RC_GPIO = 19 # gpio pin (in broadcom numbering) for reading the RC Controller's button
+PIGPIO_STEERING_MID = 1500 # PWM pulse in nanoseconds for 'straight` steering. Adjust this value if your car cannot run in a straight line.
+PIGPIO_MAX_FORWARD = 2000 # PWM pulse in nanoseconds for max forward throttle.
+PIGPIO_STOPPED_PWM = 1500 # PWM pulse in nanoseconds for zero throttle
+PIGPIO_MAX_REVERSE = 1000 # PWM pulse in nanoseconds for max reverse throttle.
+PIGPIO_SHOW_STEERING_VALUE = False
+PIGPIO_INVERT = False # rarely a controller uses an inverted pulse; if so then set to True
+PIGPIO_JITTER = 0.025 # threshold below which no signal is reported (debounce/noise rejection)
+
+Optionally, you can use the RaspberryPi to generate PWM (see Standard RC with ESC and Steering Servo) for controlling the motor speed and steering rather than using a PCA9685 board (there, see you just paid for the RC Hat!).
+Connection: +The RC hat includes two 3-pin headers compatible with the servo cables that connect to the ESC and the steering servo. Plug your car's servo into the Servo pins and the Motor Controller into the Motor pins. In all cases, make sure you plug them in the right way, noting the +,- and S (Signal) markings. Typically the black wire is "-", the red wire in the middle is "+" and the white wire is "S".
+Configuration:
+For RC output, select PWM_STEERING_THROTTLE
as your drive train type in your myconfig.py file. Uncomment the DRIVE_TRAIN_TYPE
line (remove the leading #
) and edit it as follows:
DRIVE_TRAIN_TYPE = "PWM_STEERING_THROTTLE"
+
+Then uncomment the entire PWM_STEERING_THROTTLE
configuration block and make sure steering uses "PIGPIO.BCM.13"
and throttle uses "PIGPIO.BCM.18"
because that is how the pins on the RC Hat are connected to the RaspberryPi 40 pin header.
PWM_STEERING_THROTTLE = {
+ "PWM_STEERING_PIN": "PIGPIO.BCM.13", # PWM output pin for steering servo
+ "PWM_STEERING_SCALE": 1.0, # used to compensate for PWM frequency differents from 60hz; NOT for adjusting steering range
+ "PWM_STEERING_INVERTED": False, # True if hardware requires an inverted PWM pulse
+ "PWM_THROTTLE_PIN": "PIGPIO.BCM.18", # PWM output pin for ESC
+ "PWM_THROTTLE_SCALE": 1.0, # used to compensate for PWM frequence differences from 60hz; NOT for increasing/limiting speed
+ "PWM_THROTTLE_INVERTED": False, # True if hardware requires an inverted PWM pulse
+ "STEERING_LEFT_PWM": 400, #pwm value for full left steering
+ "STEERING_RIGHT_PWM": 200, #pwm value for full right steering
+ "THROTTLE_FORWARD_PWM": 400, #pwm value for max forward throttle
+ "THROTTLE_STOPPED_PWM": 300, #pwm value for no movement
+ "THROTTLE_REVERSE_PWM": 220, #pwm value for max reverse throttle
+}
+
+After configuring the RC hat to read the RC controller and optionally control the ESC and steering servo you should do the normal calibration step to figure out the correct steering and throttle PWM values for your car (and to make sure you've hooked things up correctly).
+If one channel is reversed (steering left goes right, etc), either reverse that channel on your RC transmitter (that's usually a switch or setting) or change it in the output options shown above by changing the PWM_INVERTED value for that channel to True
.
Enable the display in myconfig.py
.
# SSD1306_128_32
+USE_SSD1306_128_32 = True # Enable the SSD_1306 OLED Display
+SSD1306_128_32_I2C_BUSNUM = 1 # I2C bus number
+SSD1306_RESOLUTION = 1 # 1 = 128x32; 2 = 128x64
+
+One of the cool things about having an OLED screen is that you can show your car's IP address on startup, so you can connect to it. Instructions to set that up are here
+If you are unable to start the car, ensure that the Adafruit_SSD1306
package is installed in your virtual environment. This should automatically be installed, if you are using a recent version of donkeycar
.
pip install Adafruit_SSD1306
+
+If you're using a standard wheel encoder, you can plug it into the "Encoder" pins, then setup the encoder configuration in your myconfig.py to use the pin that is exposed by the RC hat's encoder header.
+ +This part utilize a Google Coral accelerator and a pre-trained object detection model by Coral project to perform stop sign detection. If the donkey car see a stop sign, it will override the pilot/throttle
to 0. In addition, a bounding box will be annotated to the cam/image_array
.
To use this part, you must have:
+ +Put the following lines in myconfig.py
STOP_SIGN_DETECTOR = True
+STOP_SIGN_MIN_SCORE = 0.2
+STOP_SIGN_SHOW_BOUNDING_BOX = True
+
+Follow the Coral Edge TPU get started instructions to install the necessary software. For the RaspberryPi follow the Linux instructions.
+The stop sign detector uses a pre-compiled model, so we only need the inference runtime to make this work. However, if you are creating your own model then you will need the Edge TPU Compiler on your RaspberryPi (or Linux laptop if you are training on that). Note that the compiler only runs on Linux.
+Since the pre-trained model are trained on coco, there are 80 objects that the model is able to detect. You can simply change the STOP_SIGN_CLASS_ID
in stop_sign_detector.py
to try.
Since SSD is not good at detecting small objects, the accuracy of detecting the stop sign from far away may not be good. There are some ways that we can make enhancement but this is out of the scope of this part.
+There is an issue in the Github for making this work without the Coral Edge TPU. If you get this working please submit a pull request.
+ +This is the standard donkey data store. The "data" folder is what we call a "tub"
+The following datatypes are supported.
+str
int
float
/ np.float
image_array
s and array
s (np.ndarray
)image
(jpeg / png)The Tub
is an append only format, that is optimized for reads (to speed up training models).
+It maintains indexes for records, and uses memory mapped files.
The Tub
exposes an Iterator
that can be used to read records. These iterators can be further used by Pipeline
s to do arbitrary transformations of data prior to training (for data augumentation).
from donkeycar.parts.tub_v2 import Tub
+
+# Here we define records that have a single `input` of type `int`.
+inputs = ['input']
+types = ['int']
+tub = Tub(path, inputs, types)
+
+
+
+ This part works together with a public Alexa skill that we have released. When +you say a command, the Alexa skill will forward this command to a server hosted +by us to temporarily store it. Your donkey car, installed with this part and +with proper configuration, poll our server for any new command from Alexa.
+ +Click the image below to open the video on youtube
+ +Alexa app
, navigate to Skills and GamesTo install this part, add the following lines to manage.py
, right after the
+controller
setup. In manage.py:
+if cfg.USE_ALEXA_CONTROL:
+ from donkeycar.parts.voice_control.alexa import AlexaController
+ V.add(AlexaController(ctr, cfg), threaded=True)
+
+In myconfig.py, add the following parameters:
+USE_ALEXA_CONTROL = True
+ALEXA_DEVICE_CODE = "123456"
+
+Phrases: autopilot, start autopilot
If you use this command, it is expected that the donkey car is started with a
+model. This command will set the variable mode
of the controller to local
.
Phrases: slow down, speed up, go faster, go slower
This command alters the cfg.AI_THROTTLE_MULT
variable passed from the
+constructor. Each time this command is received, the AI_THROTTLE_MULT
is
+increased/decreased by 0.05.
Note: Since this command alters AI_THROTTLE_MULT
, it won't speed up when you
+are running in user
or local_angle
mode.
Phrases: human control, user mode, stop autopilot, manual
This command will set the variable mode
of the controller to user
Phrases: report device code, what is your device code, device code
Device code is a 6-digit numeric string derived by a hash function from your +Alexa device ID. In order to distinguish commands from multiple Alexa devices, +commands sent to our server would require an identifier, which is the device code. +When donkey car poll for new command, the part will use this device code to poll +for new commands.
+Check here for our web service source code, it is open source too.
+https://github.com/robocarstore/donkeycar-alexa-backend
+Copyright (c) 2020 Robocar Ltd
+ +Most hobby grade RC cars will work fine with the electronics, but you'll need to make your own base-plate and camera +holder. To make sure the car will work with Donkey check these things.
+For more information, see Roll Your Own.
+The easiest thing to do would be to take your parts down to your local RC / hobby shop and check that the car you want +works with the parts. Here are some parts people have said work in other countries.
+You can use tape, ribbon or even rope. The most popular tracks are 4ft wide and have 2in white borders with a dashed yellow center line. The Oakland track is about 70 feet around the center line. Key race characteristics include:
+Yes. It's all python so you can run it on any system. Usually the hard part of porting Donkey will be getting the hardware working. +Here are a couple systems that people have tried or talked about.
+NVIDA TX2 - This was implemented with a webcam and used a teensy to control the motor/servos. I2c control of PCA9685 works as well.
+Pi-Zero - Yes, try following the steps for the PiB/B+. They should work for the PiZero.
+conda activate donkey
+
+~/.bashrc
to have it active each time you login.cd donkeycar
+git pull origin main
+donkey createcar --path ~/mycar --overwrite
+
+
+ This part of documentation it was left as reference to old original classic design, because it may still bring value for some Do-It-Yourself users.
+In the future this should be moved to another sections in the docs.
+If you purchased parts from the Donkey Car Store, skip to step 3.
+I printed parts in black PLA, with .3mm layer height with a .5mm nozzle and no supports. The top roll bar is designed to be printed upside down.
+Almost all 3D Printed parts will need clean up. Re-drill holes, and clean up excess plastic.
+ +In particular, clean up the slots in the side of the roll bar, as shown in the picture below:
+ +If you have an Exceed Short Course Truck, Blaze or Desert Monster watch this video
+Slide the nut into the slot in the side of the roll cage. This is not particularly easy. You may need to clean out the hole again and use a small screwdriver to push the screw in such that it lines up with the hole in the bottom of the roll cage.
+ +Once you have slid the nut in, you can attach the bottom plate. Once again, this may be tricky. I use the small screwdriver to push against the nut to keep it from spinning in the slot. Good news: you should never have to do this again.
+ +You could do this after attaching the Raspberry Pi to the bottom plate, I just think it is easier to see the parts when they are laying on the workbench. Connect the parts as you see below:
+ +For reference, below is the Raspberry Pi Pinout for reference. You will notice we connect to 3.3v, the two I2C pins (SDA and SCL) and ground:
+ +Before you start, now is a good time to insert the already flashed SD card and bench test the electronics. Once that is done, attaching the Raspberry Pi and Servo is as simple as running screws through the board into the screw bosses on the top plate. The M2.5x12mm screws should be the perfect length to go through the board, the plastic and still have room for a washer. The “cap” part of the screw should be facing up and the nut should be on the bottom of the top plate. The ethernet and USB ports should face forward. This is important as it gives you access to the SD card and makes the camera ribbon cable line up properly.
+Attach the USB battery to the underside of the printed bottom plate using cable ties or velcro.
+ +There are two versions of the donkey chassis, the newer one does not have screws, the older one does. This includes instructions for both:
+Screwless Design +The newer design is pretty simple, just slip the camera into the slot, cable end first. However, be careful not to push on the camera lens and instead press the board. +
+If you need to remove the camera the temptation is to push on the lens, instead push on the connector as is shown in these pictures.
+
+
Design with Screws
+Attaching the camera is a little tricky, the M2 screws can be screwed into the plastic but it is a little hard. I recommend drilling the holes out with a 1.5mm bit (1/16th bit in Imperial land) then pre threading them with the screws before putting the camera on. It is only necessary to put two screws in.
+++Sometimes using the two top screw holes can result in a short. Put screws in the bottom two holes.
+
Before using the car, remove the plastic film from the camera lens.
+ +It is easy to put the camera cable in the wrong way so look at these photos and make sure the cable is put in properly. There are loads of tutorials on youtube if you are not used to this.
+ +*** Note if you have a Desert Monster Chassis see 7B section below ***
+The final steps are straightforward. First attach the roll bar assembly to the car. This is done using the same pins that came with the vehicle.
+ +Second run the servo cables up to the car. The throttle cable runs to channel 0 on the servo controller and steering is channel 1.
+ +Now you are done with the hardware!!
+The Desert monster does not have the same set up for holding the body on the car and needs two adapters mentioned above. To attach the adapters you must first remove the existing adapter from the chassis and screw on the custom adapter with the same screws as is shown in this photo:
+ +Once this is done, go back to step 7
+Congrats! Now to get your get your car moving, see the software instructions section.
+ + +The donkey
command is created when you install the donkeycar Python package. This is a Python script that adds some important functionality. The operations here are vehicle independent, and should work on any hardware configuration.
This command creates a new dir which will contain the files needed to run and train your robot.
+Usage:
+donkey createcar --path <dir> [--overwrite] [--template <donkey2>]
+
+--path
as the destination dir to create. If .py
files exist there, it will not overwrite them, unless the optional --overwrite
is used.--overwrite
will update the files in the destination directory except the myconfig.py
. This is useful if you have update donkeycar and you want those changes reflected in you mycar
folder but you don't want to have to recreate your myconfig.py
.--template
will specify the template file to start from. For a list of templates, see the donkeycar/templates
dir. This source template will be copied over the manage.py
for the user. Common templates are:--template=complete
: the Deep Learning Autopilot--template=path_follow
: the Path Follow Autopilot--template=cv_control
: the Computer Vision AutopilotThis command attempts to locate your car on the local network using nmap.
+Usage:
+donkey findcar
+
+sudo apt install nmap
+
+This command allows you to manually enter values to interactively set the PWM values and experiment with how your robot responds. +See also more information.
+Usage:
+donkey calibrate --channel <0-15 channel id>
+
+--channel
Ctrl + C
to exitOpens a web server to delete bad data from a tub.
+Usage:
+donkey tubclean <folder containing tubs>
+
+Ctrl + C
to exitNote: This section only applies to version >= 4.1 +This command trains the model. There is more detail in Deep Learning Autopilot.
+donkey train --tub=<tub_path> [--config=<config.py>] [--model=<model path>] [--type=(linear|categorical|inferred)] [--transfer=<transfer model path>]
+
+--tub
datastore. You may specify more than one tub using a comma separated list --tub=foo/data,bar/data
or just leaving spaces like --tub foo/data bar/data
.--config
path (optionally)--model
. Auto-generates a model name if omitted. Note: There was a regression in version 4.2 where you only had to provide the model name in the model argument, like --model mypilot.h5
. This got resolved in version 4.2.1. Please update to that version.--type
--transfer
TRAIN_FILTER
in the myconfig.py
file. For example: def filter_record(record):
+ return record.underlying['user/throttle'] > 0
+
+TRAIN_FILTER = filter_record
+
+only uses records with positive throttle in training.
+donkey train --tub=<tub_path> [--config=<config.py>] [--model=<model path>] [--type=(linear|categorical|inferred|rnn|imu|behavior|localizer|3d)] [--transfer=<transfer model path>]
+
+In addition, a Tflite model is automatically generated in training. This can be suppressed by setting CREATE_TF_LITE = False
in your config. Also Tensorrt models can now be generated. To do so, you set CREATE_TENSOR_RT = True
.
createcar
command still creates a train.py
file for backward compatibility, but it's not required for training.This command allows you to create a movie file from the images in a Tub.
+Usage:
+donkey makemovie --tub=<tub_path> [--out=<tub_movie.mp4>] [--config=<config.py>] [--model=<model path>] [--model_type=(linear|categorical|inferred|rnn|imu|behavior|localizer|3d)] [--start=0] [--end=-1] [--scale=2] [--salient]
+
+--tub
dir path given--out
. Codec is inferred from file extension. Default: tub_movie.mp4
config.py
other than default: config.py
--salient
will overlay a visualization of which pixels excited the NN the most--start
and/or --end
can specify a range of frame numbers to use.This command allows you to plot steering and throttle against predictions coming from a trained model.
+Usage:
+donkey tubplot --tub=<tub_path> --model=<model_path> [--limit=<end_index>] [--type=<model_type>]
+
+~/mycar
dir--limit=<end_index>
will use all records up to that index, defaults to 1000.--type=<model_type>
will use a different model type than the DEFAULT_MODEL_TYPE
Note: Requires version >= 4.3
+This command allows you to plot tub data (usually steering and throttle) as a histogram.
+Usage:
+donkey tubhist --tub=<tub_path> --record=<record_name> --out=<output_filename>
+
+~/mycar
dir--record=<record_name>
will only show the histogram of a certain data series, for example "user/throttle"--out=<output_filename>
saves histogram under that name, otherwise the name is auto-generated from the tub pathThis command line wizard will walk you through the steps to create a custom/customized controller.
+Usage:
+donkey createjs
+
+~/mycar
dirjstest
can be useful here. Installed via: sudo apt install joystick
You must pass this utility the path to your controller's device. Typically this is /dev/input/js0
However, it if is not, you must find the correct device path and provide it to the utility. You will need this for the createjs command as well.donkey createjs
and it will create a file named my_joystick.py in your ~/mycar
folder, next to your manage.pyCONTROLLER_TYPE="custom"
to use your my_joystick.py controllerShows feature maps of the provided image for each filter in each of the convolutional layers in the model provided. Debugging tool to visualize how well feature extraction is performing.
+Usage:
+donkey cnnactivations [--tub=<data_path>] [--model=<path to model>]
+
+This will open a figure for each Conv2d
layer in the model.
Example:
+donkey cnnactivations --model models/model.h5 --image data/tub/1_cam-image_array_.jpg
+
+Note: This is only available in donkeycar >= 4.3.1.
+This lists the models that are stored in models/database.json
. Displays information
+like model type, model name, tubs used in training, transfer model and a comment if
+--comment
was used in training or the model was trained in the UI.
Usage:
+donkey models [--group]
+
+~/mycar
directory--group
flag is given, then the tub path info is combined into groups,
+if different models used different tubs. Useful, if you use multiple tubs, and the models
+have used different tub combinations because it compresses the output information. pandas
first if you want to run it on the car Note: This section only applies to version >= 4.2.0
+Usage:
+donkey ui
+
+This opens a UI to analyse tub data supporting following features:
+The UI is an alternative to the web based donkey tubclean
.
A full documentation of the UI is here.
+ +Launch the Donkey graphical training interface by entering donkey ui
in the command line. This works on Linux, Mac, and Windows, although if you're on Windows it's recommended that you use WSL (Windows Subsystem for Linux) running Ubuntu 20 instead to get full functionality.
The Donkey UI currently contains four screens supporting the following workflows:
+The tub manager - a replacement for the web-based application launched through donkey tubclean
The trainer - a UI based alternative to train the pilot. Note, for longer trainings containing larger tubs or batches it is recommended to perform these in the shell using the donkey train
command. The UI based training is geared towards an experimental and rapid analysis cycle consisting of:
The pilot arena - here you can test two pilots' performance against each other.
+Note: Under linux the app depends on xclip
, if this is not installed, then please run:
sudo apt-get install xclip
+
+In the tub manager screen you have to select the car directory that contains the config file myconfig.py
first, using the Load car directory
button. Then select the tub you want to be working with using Load tub
, the tub needs to be inside the car directory. The application remembers the last loaded config and tub.
The drop-down menu Add/remove' in the data panel to the left of the image allows to select the record fields, like
user/angle,
user/throttle`, etc.
Note: if your tub contains more data than the standard user/angle
, user/throttle
and you want the progress bars to correctly show the values of these fields, you need to add an entry into the .donkeyrc
file in your home directory. This file is automatically created by the Donkey UI app. Here is an example:
field_mapping:
+- centered: true
+ field: car/accel
+ max_value_id: IMU_ACCEL_NORM
+
+This data entry into the field_mapping
list contains the name of the tub field, a switch, if the data is centered around 0 and the name of the maximum value of that data field which has to be provided in the myconfig.py
file. For example, the data above represents the IMU acceleration of the IMU6050 which ranges between +/- 2g, i.e. ~ +/-20 m/s2. With an IMU_ACCEL_NORM of 20 the progress bar can display these values. Therefore, the myconfig.py
should contain:
IMU_ACCEL_NORM = 20
+
+Note: Vectors, i.e. list / arrays are being decomposed by the UI into their components automatically.
+Here is an example of a tub that has car/accel
and car/gyro
arrays that hold IMU data, as well as car/distance
and car/m_in_lap
. The first two show a progress bar because there is a corresponding entry in the field_mapping
list as explained above.
+
The control panel allows moving forward and backward in single steps using <, > and scrolling continuously with <<, >>. These buttons are also linked to the keyboard keys < left >, < right >, < space >.
+To delete unwanted records press Set left
/ Set right
buttons to determine the range and hit Delete
to remove such records from training. To see the impact on the current tub press Reload tub
. If you want to resurrect accidentally deleted records, just choose a left/right value outside the deleted range and press Restore
.
Note: The left/right values are invertible, i.e. left > right operates on all records outside [left, right).
+In the filter section you can suppress records, say you want to restrict the next training on only right curves, then add user/angle > 0
to select those records.
+Note: the filter on only for display in the tub manager. If you want to apply this in training you have to write the predicate as explained in utility
The lower panel contains a graph with the currently selected data from the data panel. If nothing is selected, all fields from the record are displayed. The display scales the data between the minimum and maximum value of each record field, hence there are no absolute measurements possible. For more advanced data graphing capabilities, press Browser Graph
which opens a plotly history graph in the web browser.
The trainer screen allows to train a model to the tub data. In the Overwrite config
section you can set any config parameter by typing an updated value into the text field on the right and hitting return.
To train a pilot use the model type dropdown, enter a comment and hit Train
. After training, the pilot will appear in the pilot database which is shown underneath. You can also choose a transfer model if you don't want to train from scratch. Note, tensorflow saves the model with the optimiser state, so training will commence where it stopped in the saved state.
Pilots might be trained on multiple tubs, this is currently not supported in the trainer. However, if multiple tubs are passed in donkey train
then these will show in the database too. In order to not clutter up the view and group different tubs, you can use the Group multiple tubs
button to group all tub groups of two and more, and show a group alias instead. The group alias mapping is shown in the lower area of the window then.
Here you can benchmark two pilots against each other. Use this panel to test if changes in the training through optimiser parameters or model types, or through deletion of certain records, or augmentations to images have made the pilot better or worse. The last selected pilots will be remembered in the app.
+Choose a pilot by selecting the Model type
and loading the keras model using the file chooser by pressing Choose pilot
. The control panel is the same as in the tub manager. The lower right data panel shows the tub record's data. You can select the throttle field as some folks train on car speed instead of throttle values. In such case, the corresponding field name must be added into the .donkeyrc
file in the section, see an example here.
user_pilot_map:
+ car/speed: pilot/speed
+ user/angle: pilot/angle
+ user/throttle: pilot/throttle
+
+The 'user/angle' and 'user/throttle' mappings are automatically loaded by the app. In order to show the variable car/speed
and compare it to the AI produced pilot/speed
the map has to contain the corresponding entry.
Under the two pilots there are sliders with pre-defined image augmentations, here Brightness
and Blur
. You can mix brightness and blur values into the images and compare how well the pilots are reacting to such a modification of the testing data. Press the buttons to activate the sliders for enabling this feature.
The application will remember the last two selected pilots.
+Note: This screen will only work on linux / OSX as it makes use of ssh
and rsync
in the background. SSH needs to be configured to allow login from the PC to the car without password. This can be done by running on the PC:
ssh-keygen
+
+When being asked for a passphrase just hit <return>. This creates a public and private key in your ./ssh
directory. Then copy the key over to the car with the command below. Here we assume your car's hostname is donkeypi
- otherwise replace it with the corresponding hostname.
ssh-copy-id -i ~/.ssh/id_rsa.pub pi@donkeypi.local
+
+Log into your car using:
+ssh pi@donkeypi.local
+
+If SSH asks you if that host should be added to the list of known hosts, hit <return> and you are done. From now on, you can ssh into the car without being prompted for the password again. The login-free setup is required for the screen to work.
+PI_USERNAME
and PI_HOSTNAME
are set to your car user's username and the hostname of the car.With the car connector you can transfer the tub data from the car to your PC and transfer the pilots back to the car.
+Under the Car directory
enter the car folder and hit return. This should populate the Select tub
drop down. Most likely you want to select the data/
directory but you might have tubs in subfolders. In that case use ~/mycar/data
in the Car directory
, select the tub you want to pull and enable Create new folder
on the button. This will copy a tub on the car like ~/mycar/data/tub_21-04-09_11
into the same location on your PC. Without the Create new folder
it would copy the content of the car's tub folder into ~/mycar/data
of your PC, possibly overwriting other tub data that might be there.
Press Pull tub data/
to copy the tub from the car
Send pilots
to sync your local models/
folder with the models/
folder on the car. This command syncs all pilots that are locally stored.Drive car
section you can start the car and also select a model for autonomous driving. After starting you have to use the controller, either web or joystick as usual.myconfig.py
file.~/.donkeyrc
file to the kivy internal settings.You can find a video tutorial for the UI below
+ + +