INDI library (https://indilib.org/) is an open source software to control astronomical equipment.
Raspberry Pi cameras allow the amateur astronomer to make astonishing pictures with small budget. Especially the Raspberry Pi HQ camera can compete with expensive astro cameras.
This project implements a Raspberry Pi camera driver for INDI. It is based on the new camera framework "libcamera" (https://github.com/raspberrypi/libcamera) which is already part of many Raspberry Pi operating systems.
The driver is made and optimized to run on a Raspberry Pi Zero wih HQ camera. Ofcourse it will also run on a more capable Raspberry Pi.
The "indi_pylibcamera" may support all cameras supported by "libcamera". But not all cameras will provide image data in the required formats (raw Bayer or at least RGB). So it is not guaranteed that the driver will work with all cameras you can connect to a Raspberry Pi.
- Python 3 and some libraries:
sudo apt-get install python3 python3-lxml python3-astropy python3-picamera2
- Libcamera (if not already installed:)
sudo apt-get install libcamera
. You can test libcamera and the support for your camera with:libcamera-hello --list-cameras
- Install INDI core library. If there is no pre-compiled package for your hardware you will need to compile it
by yourself. Instructions can be found here: https://github.com/indilib/indi. A Raspberry Pi Zero does not
have enough RAM to compile with 4 threads in parallel: you need to do
make -j1
instead ofmake -j4
. Finally, after installation, you need to have a working INDI server:indiserver -v indi_simulator_telescope
Currently, the indi_pylibcamera
driver does not has a setup or installation tool. Just copy the files in a folder.
The script install.sh
installs the requirements and creates a file indi_pylibcamera.xml
in /usr/share/indi
to get the driver listed in EKOS profile editor.
Furthermore it creates a soft link /usr/bin/indi_pylibcamera
so that the driver can be used without prior changing to its directory.
The script will ask you for sudo premissions to create the file in /usr/bin
.
Usage:
install.sh
You can start the INDI server with indiserver -v ./indi_pylibcamer.py
after changing to the folder where you stored
indi_pylibcamera.py
. When the server is running you can connect to the server from another computer with an INDI
client (for instance KStars/EKOS).
The driver uses configuration files to set global parameter. If environment variable INDI_PYLIBCAMERA_CONFIG_PATH
exists the file $INDI_PYLIBCAMERA_CONFIG_PATH/indi_pylibcamera.ini
is loaded. Otherwise, it tries to load
$HOME/.indi_pylibcamera/indi_pylibcamera.ini
.
The configuration file must have the section [driver]
. The most important keys are:
DeviceName
(string): INDI name of the device. This allows to distinguish indi_pylibcamera devices in your setup. For instance you can have one Raspberry Pi with HQ camera as main camera for taking photos and a second Raspberry Pi with a V1 camera for auto guiding.SendTimeStamps
(yes
,no
,on
,off
,true
,false
,1
,0
): Add a timestamp to the messages send from the device to the client. Such timestamps are needed in very seldom cases only, and usually it is okay to set this tono
. If you really need timestamps make sure that the system clock is correct.force_UnitCellSize_X
,force_UnitCellSize_Y
andforce_Rotation
: Some cameras are not fully supported by libcamera and do not provide all needed information. The configuration file allows to force pixel size and Bayer pattern rotation for such cameras.LoggingLevel
: The driver has buttons to set the logging level. But sometimes you need a higher logging level right at the beginning of the driver initialization. This can be done here in the INI file.DoSnooping
: The INDI protocol allows a driver to ask other drivers for information. This is called "snooping". The indi_pylibcamera driver uses this feature to get observer location, telescope information and telescope direction from the mount driver. It writes these information as metadata in the FITS images. This function got newly implemented and may make trouble in some setups. With theDoSnooping
you can disable this function.
There are more settings, mostly to support debugging.
An example for a configuration file can be found in this repository.
When killing the indiserver sometimes the driver process continues to run. You can see this with:
ps ax | grep indi
If you get python3 ././indi_pylibcamera.py
in the output the driver process is still running. In that case you must
kill the driver process manually before you restart the indiserver. Otherwise, you will get a libcamera error
when connecting to the camera.
The driver is made as generic as possible by using the camera information provided by libcamera. For instance the raw modes and frame sizes selectable in the driver are coming from libcamera. Unfortunately some important information is not provided by libcamera:
- Some cameras have columns or rows filled with 0 or "garbage". These can disturb postprocessing of frames. For instance an automated color/brightness adjustment can get confused by these 0s.
- Libcamera creates some raw modes by cropping, and others by binning (or subsampling, this is not sure) of frames. Binning (and subsampling) influences the viewing angle of the pixel. An INDI CCD driver must provide the used binning to allow client software to calculate the image viewing angle.
To work around this the driver makes a special handling for the cameras listed below. Due to the removing of zero-filled columns/rows the image frame size will be smaller than stated on the raw mode name.
Libcamera provides 4 raw modes for this camera, some made by binning (or subsampling) and all with 0-filled columns:
- 4056x3040 BGGR 12bit: provided frame size 4064x3040, no binning, has 8 zero-filled columns on its right, final image size is 4056x3040
- 2028x1520 BGGR 12bit: provided frame size 2032x1520, 2x2 binning, has 6 zero-filled columns on its right (8 get removed to avoid 1/2 Bayer pattern), final image size is 2024x1520
- 2028x1080 BGGR 12bit: provided frame size 2032x1080, 2x2 binning, has 6 zero-filled columns on its right (8 get removed to avoid 1/2 Bayer pattern), final image size is 2024x1080
- 1332x990 BGGR 10bit: provided frame size 1344x990, 2x2 binning, has 12 zero-filled columns on its right, final image size is 1332x990
Maximum exposure time is > 5 minutes.
This camera does not add zero-filled columns. But libcamera uses 3 binning modes. Maximum exposure time is 1 sec.
- 2592x1944 GBRG 10bit: provided frame size 2592x1944, no binning, no garbage columns, final image size is 2592x1944
- 1920x1080 GBRG 10bit: provided frame size 1920x1080, no binning, no garbage columns, final image size is 1920x1080
- 1296x972 GBRG 10bit: provided frame size 1296x972, 2x2 binning, no garbage columns, final image size is 1296x972
- 640x480 GBRG 10bit: provided frame size 640x480, 4x4 binning, no garbage columns, final image size is 640x480
This camera has auto-focus capabilities which are not supported by this driver. Maximum exposure time is 1.7 sec.
- 4608x2592 BGGR 10bit: provided frame size 4608x2592, no binning, no garbage columns, final image size is 4608x2592
- 2304x1296 BGGR 10bit: provided frame size 2304x1296, 2x2 binning, no garbage columns, final image size is 2304x1296
- 1536x864 BGGR 10bit: provided frame size 1536x864, 2x2 binning, no garbage columns, final image size is 1536x864
In case you have trouble, or you see unexpected behavior it will help debugging when you give more information about your camera. Please run:
./print_camera_information.py > MyCam.txt
and send the generated "MyCam.txt" file.
Furthermore, send one raw image for each available raw mode. Make pictures of a terrestrial object with red, green and blue areas. Do not change camera position between taking these pictures. It must be possible to measure and compare object dimensions.
The indi_pylibcamera driver uses snooping to get information from the mount driver. This information is used to add more metadata to the FITS images, similar to this:
FOCALLEN= 2.000E+03 / Focal Length (mm)
APTDIA = 2.000E+02 / Telescope diameter (mm)
SCALE = 1.598825E-01 / arcsecs per pixel
SITELAT = 5.105000E+01 / Latitude of the imaging site in degrees
SITELONG= 1.375000E+01 / Longitude of the imaging site in degrees
AIRMASS = 1.643007E+00 / Airmass
OBJCTAZ = 1.121091E+02 / Azimuth of center of image in Degrees
OBJCTALT= 3.744145E+01 / Altitude of center of image in Degrees
OBJCTRA = ' 4 36 07.37' / Object J2000 RA in Hours
OBJCTDEC= '16 30 26.02' / Object J2000 DEC in Degrees
RA = 6.903072E+01 / Object J2000 RA in Degrees
DEC = 1.650723E+01 / Object J2000 DEC in Degrees
PIERSIDE= 'WEST ' / West, looking East
EQUINOX = 2000 / Equinox
DATE-OBS= '2023-04-05T11:27:53.655' / UTC start date of observation
This function is newly implemented. It is tested with the indi_simulator_telescope and the indi_synscan_telescope mount drivers. If you get trouble with this function you can disable snooping in the INI file.
A correct system time on you Raspberry Pi is absolutely needed for the calculation of the metadata. The Raspberry Pi does not have a battery powered realtime clock. It adjusts its system time from a time-server in the internet. If your Pi does not have internet access you will need to take care for setting the date and time. For instance, you can install a realtime clock or a GPS hardware. You can also copy date and time from one Linux computer (or Raspberry Pi) to another with:
ssh -t YourUserName@YourPiName sudo date --set=`date -Iseconds`
The driver uses "astropy" (https://www.astropy.org/) for coordinate transformations. When processing of the first image you make the "astropy" library needs a few seconds for initialization. This will not happen anymore for the next images.
Snooping takes telescope focal length and aperture from the mount driver. It picks there the values of the main telescope. When your camera is on a different optic you can force aperture and focal length with the "Scope" setting in the "Options" tap of the indi_pylibcamera driver.
- The maximum exposure time of the V1 camera is about 1 second. This limitation is caused by libcamera and the kernel driver. The indi_pylibcamera can not work around this.
- Libcamera reports a maximum exposure time for HQ camera of 694.4 seconds. But when trying an exposure of 690 seconds the capture function already returns after 40 seconds. Likely this maximum exposure time is wrong. A test with 600 seconds exposure time was successful.
- Libcamera reports a higher maximum value for analogue gain than expected. The analogue gain is implemented by hardware and has therefore well-defined restrictions. It is not clear if the reported higher maximum analogue gain is correct.