Astro-Modifikation einer Canon EOS 2000D
(aka EOS 1500D, Rebel T7, Kiss X90)
mit Kühlung, Intervall-Timer, Shutter-Emulator
und Bluetooth-Fernsteuerung
Hauptzweck dieser Modifikation war, herauszufinden, wie effektiv sich die Sensoren der neueren Canon DSLR/DSLM Modelle, bei denen der Sensor ohne einen Zwischenraum flächig auf eine Platine gelötet ist, kühlen lassen.
Dabei wollte ich den Kühler möglichst klein halten, auch wenn sich dadurch extreme Kühlung, wie z.B. 20°C unter Umgebung, nicht erreichen lässt. Das ist aber kein gravierender Nachteil, wie die Dunkelströme, die ich mit moderater Kühlung bekommen habe, zeigen. 

Ich baue aus meinen Astro-Kameras als erstes immer gleich Shutter und Spiegel aus und ersetze sie durch einen kleinen Arduino, der der Kamera die von ihr erwarteten elektrischen Signal liefert, damit sie keinen Fehler meldet. Das hat unter anderem den Vorteil, dass kein Verschleiß mehr auftritt, die Kamera völlig lautlos arbeitet und der Spiegel das Bild nicht mehr verschattet.

Wenn dann eh schon ein Arduino in der Kamera ist, ist es kein großer Aufwand, ihn mit einem Bluetooth-Modul, das nur wenig Platz braucht, zu versehen. Damit kann man dann nicht nur die Regelung der Kühlung bedienen, sondern auch noch einen Intervall-Timer steuern, für den der Arduino natürlich an den Auslöser angeschlossen werden muss. Das erspart den externen Intervall-Timer für die Modelle, für die es kein Magic Lantern gibt und ist besonders nützlich für Kameras, die keinen Kabelauslöseranschluss haben, wie die EOS M50, in die ich diese Kombination zuerst installiert hatte.

Das LPF2 Farbkorrekturfilter vor dem Sensor wird dann natürlich, falls noch nicht geschehen, auch gleich entfernt und ebenso der Sucher, der ohne Spiegel sowieso keine Funktion mehr hat.

Die Kosten des Ganzen sind sehr überschaubar: Für Arduino, Bluetooth-Modul, Spannungsregler und Peltier-Element braucht man 10 bis 15 €, alles zusammen mit Kühlkörper und Lüfter kommt dann auf unter 20 €.
Und so sieht die Kamera inklusive Kühlung dann aus. Dank der günstigen Lage des Sensoranschlusses konnte ich Peltier, Kühlkörper und Lüfter im Sucherraum unterbringen, nachdem ich den Blitz entfernt und das Top noch etwas aufgeschnitten hatte. Es gibt auch keinen separaten Stromanschluss für die Kühlung, die wird von der Kamera mitversorgt. Für eine Stunde oder so ginge das sogar mit dem Akku, aber es ist definitiv besser, alles mit einem handelsüblichen Netzgerät und Akku-Dummy zu versorgen.
Insgesamt wiegt die Kamera jetzt gerade mal 390 Gramm.

Von oben sieht man gut, wie das Top der Kamera ausgeschnitten ist, damit der Radiallüfter reinpasst. Ein etwas kleinerer Lüfter (30 statt 40 mm) hätte es auch getan und dann wäre nur die kleine Abluftöffnung notwendig gewesen. Im Moment habe ich die nicht benötigten Teile der Öffnung provisorisch mit Klebeband verschlossen. Da werde ich noch die Blitzabdeckung passend zuschneiden, dann sieht das wieder besser aus.
Für die Zuluft habe ich vorne und hinten noch einige Löcher in das Gehäuse gebohrt.
Hier ist der Aufbau des Kühlers zu sehen:
Ein Kupferblech mit einer Aussparung für das Sensorkabel führt nach oben und dann nach vorn in den Bereich, in dem der Sucher untergebracht war. Den Wärmekontakt zwischen Kupferblech und Sensorplatine übernimmt ein dickes Wärmeleit-Pad, das den Zwischenraum gut ausfüllen sollte. Es ist wichtig ein Pad zu verwenden, nicht Wärmeleitpaste, die eine viel schlechtere Wärmeleitung hat. Außerdem ist es keine solche Sauerei.
Zur Sicherheit ist das Kupferblech mit einem dünnen Paketband gegen elektrischen Kontakt mit dem Sensor isoliert. Das Kupfer ist nur 0.3mm dick, das reicht bei der geringen Wärmemenge, die transportiert werden muss, aus. Überhaupt ist es sehr wichtig, die Wärmemenge möglichst gering zu halten, dann reicht auch für das Peltier-Element eine kleine Leistung aus. Ich habe einen TEC 04901 gewählt, ein recht 'schwaches' Peltier, das maximal einen Strom von 1A aufnimmt. Seine Kühlleistung ist für den gewünschten Zweck aber ausreichend. 
Essentiell ist dabei eine sehr gute thermische Isolation des Kupferblechs nach außen, besonders auch um das Kabel zum Mainboard herum. PU-Schaum und dünne Polsterfolie sind dafür gut geeignet.

Oben auf dem Peltierelement sitzt jetzt der auf 35x45 mm² zugeschnittene Kühlkörper, auf den dann ein Radial-Lüfter aufgesetzt wird. Eine dünne Polsterfolie zwischen Lüfter und Kühlkörper dämpft eventuelle Vibrationen des Lüfters und ist so zugeschnitten, dass die von vorn und hinten angesaugte Luft den Kühlkörper möglichst effektiv durchströmt.

Die Versorgung des Peltier-Elements übernimmt einer der beliebten kleinen Mini360 Bucks, den ich hier anstelle des Blitzkondensators auf das Powerboard der 2000D gelötet habe. Er wird direkt vom Batterieanschluss mit 7.2 V gespeist. Da diese Spannung permanent anliegt, wird er über den Enable-Eingang an Pin 7 des ICs deaktiviert, wenn die Kamera abgeschaltet ist. Neben dem Spannungstrimmer ist noch ein Widerstand angelötet, der die maximale Ausgangsspannung begrenzt. An den wird dann die geglättete Steuerspannung von einem PWM-Ausgang des Arduino angeschlossen, so dass der die Kühlleistung regeln kann.
Ein zweiter Mini360 unter dem ersten regelt die Spannung für den 5V-Lüfter auf etwa 3V herunter, so dass der dann ohne viel Lärm und Vibrationen arbeitet.
Elektronik

Ein Arduino Pro Mini bildet den Kern der gesamten Steuerung. Mit seinen Datenpins D2-D4 und D6-D8 arbeitet er an der Emulation der Shuttersignale für die 2000D, von der er auch seine Versorgungsspannung von 3.3V erhält, ebenso wie das HC-05 Bluetooth-Modul, das er über D10 und D11 anbindet. D10 und D11 bilden dazu eine serielle Schnittstelle in Software nach. Man kann das Bluetooth-Modul stattdessen auch mit der eingebauten seriellen Schnittstelle über D0 und D1 anbinden, muss es dann aber immer abhängen, wenn man dem Arduino einen neuen Sketch hochladen will.
D9 ist mit dem im Bild mit 'Shoot' bezeichneten Anschluss der Kabelauslöser-Buchse verbunden, die bei der 2000D auf der Hauptplatine sitzt. Damit kann der Arduino bzw. der Intervalltimer die Kamera auslösen.

D13 geht zu den Enable-Eingängen der beiden Mini360, um die Kühlung ein- und auszuschalten und automatisch  zu deaktivieren, wenn die Kamera abgeschaltet ist. Damit die LED des Arduino, die ebenfalls an D13 hängt, nicht leuchtet, wenn die Kühlung läuft, muss sie ausgelötet oder ausgebrochen werden.

D5 wird als PWM-Ausgang verwendet. Die daran angeschlossene Kombination aus zwei Widerständen und einem Kondensator glättet das Rechtecksignal, das dann in die Spannungsregelung des Mini360, der das Peltier versorgt, eingespeist wird und regelt damit die Kühlleistung. Der Regelbereich beträgt etwa 1V bis 4V.
Die zusätzlich an die Mini360 gelöteten Widerstände begrenzen deren Ausgangsspannung auf 5.2V (Peltier) bzw. 4.6V (Lüfter). Sie sind nicht unbedingt notwendig.

Um das Beschlagen des Sensors zu verhindern, liegen im Spiegelkasten noch vier Widerstände mit je 5.1Ω, die von der Peltier-Spannung gespeist werden und damit proportional zur Kühlleistung heizen. Sie können zusätzlich dadurch geregelt werden, dass sie über D12 und einen MOSFET gataktet bestromt werden.

Die Temperatur (des Kupfer-Kühlblechs) wird mit einem NTC Thermistor bestimmt. Dazu wird seine Spannung am A0-Eingang des Arduino gemessen. Der zweite Widerstand an A0 sollte den gleichen Wert haben wie der NTC bei 25°C, das vereinfacht die Berechnung der Temperatur aus der gemessenen Spannung. Ich habe hier eine 10kΩ Typ verwendet, es gehen aber z.B. auch 100kΩ.

Wichtig: Damit ein Sketch hochgeladen werden kann, muss die Kamera eingeschaltet sein, da sie ja den Arduino  versorgt. Die Versorgung über USB / Serial Breakout darf dann nicht parallel zur Vcc der Kamera geschaltet werden!
Bluetooth-Steuerung

Die Verschaltung mit dem Bluetooth-Modul ist oben ja schon gezeigt. Die Frage ist jetzt, wie man damit am besten kommuniziert.
Zum Glück gibt es dafür schon zahlreiche fertige Lösungen, das heist Apps, die auf dem Handy oder Tablet mit Bluetooth-Geräten Daten austauschen. Man könnte sogar mit überschaubarem Aufwand eine eigene App dafür zusammenstellen, aber ich habe mich erst mal mit dem "Serial Bluetooth Terminal" begnügt, wobei 'begnügt' eigentlich nicht das richtige Wort ist, denn das ist eine sehr schöne App, die genau das tut, was ich für den Zweck brauche. Der Bluetooth-Code auf der Arduino-Seite ist Teil des ShutterEmuEOS Programms. Er ist für das Bluetooth Terminal ausgelegt und kann bei Bedarf leicht angepasst werden.
Dieser Screen-Shot des Bluetooth-Terminals zeigt in der oberen Hälfte den Output der Hilfefunktion, die aktuellen Einstellungen und den Status der Kamera.

Im unteren Teil bietet das Bluetooth-Terminal ein konfigurierbares 'Tastenfeld'. Hinter jeder Taste steht ein Makro, das man passend einrichten kann. Hier sind die Makros so gestaltet, dass sie jeweils ein Zeichen senden, um die wichtigsten Befehle einfach auszuführen. Einige Befehle erwarten eine Ziffernfolge, die vor dem Befehlszeichen eingegeben wird.

Die Tastenkonfiguration kann man auch in eine Datei exportieren, um sie auf dem Rechner zu bearbeiten oder auf anderen Geräten zu importieren. Eine Kopie davon liegt dem ShutterEmuEOS Programm bei.
Im linken Teil des Tastenfelds befinden sich die Befehle für den Intervall-Timer, der Bildserien mit frei wählbarer Belichtungszeit (und Pausen) auslöst. Man kann die Kamera dazu sowohl im Bulb-Mode als auch mit allen anderen Modi betreiben, in denen die Kamera die Belichtungszeit vorgibt, zum Beispiel um Zeitrafferreihen aufzunehmen.

Unten gibt es dann noch die Taste 'Kühlung', mit der die Anweisungen zur Einstellung der Kühlung abgesetzt werden. So schaltet zum Beispiel die Zeichenfolge '1K', die mit den Tasten '1' und 'Kühlung' gesendet wird, die Kühlung auf niedrigste Stufe, während '8K' den PID-Regler aktiviert, der die Temperatur konstant hält.


Die Baudrate für die Kommunikation zwischen Bluetooth-Modul und Arduino ist auf 9600 voreingestellt, das ist das, was auch für einen HC-05 vorkonfiguriert ist. Das ist zwar nicht berauschend, aber es reicht aus. Was ich am HC-05 noch umkonfiguriere, ist der Name, unter dem er sich meldet, also z.B. "2000Daci".
Die aktuellen Einstellungen gehen verloren, wenn die Kamera ausgeschaltet wird. Man kann sie jedoch (mit dem E-Befehl) im EEPROM des Arduino speichern, von wo sie bei einem Neustart wieder geladen werden.

Intervalltimer

Der Intervalltimer kann die Kamera periodisch auslösen. Für Langzeit-Belichtungen wird an der Kamera der Bulb-Mode gewählt und der Intervalltimer bestimmt die Belichtungszeit, die mit '+' oder '-' aus einer Reihe vordefinierter Werte ausgewählt oder in Sekunden oder Minuten eingegeben wird.
Der kleinste der vordefinierten Werte, der durch wiederholte Eingabe von '-' gewählt wird, ist dazu gedacht, die Belichtungszeit mit der Kamera vorzugegeben. In diesem "Schnelle Auslösungen" Modus löst der Intervalltimer zehn mal pro Sekunde aus. Die Kamera ignoriert verfrühte Auslösungen und startet das nächste Bild erst wenn sie wieder bereit ist.
 
Durch die Wahl einer verlängerten Pause zwischen den Auslösungen kann die Belichtungszeit unabhängig von der Verzögerung durch die Bildspeicherung exakt definiert werden. 
Voreingestellt ist eine Pause von 0.2 Sekunden, die aber von der Kamera, wenn sie noch nicht bereit ist, verlängert wird, und zwar auf Kosten der folgenden Belichtung.
Bei sehr kurzen Pausen startet unter Umständen nur jede zweite Auslösung! 
Mit langen Pausen können auch Timelapse-Reihen aufgenommen werden. Dazu muss für den Timer die Belichtungszeit auf mindestens eine Sekunde gestellt werden, weil der "Schnelle Auslösungen" Modus keine langen Pausen macht. 
Wenn die Kamera die Belichtungszeit vorgeben soll, können Timelapse-Reihen auch einfach durch die Wahl einer entsprechend langen Belichtungszeit im Intervalltimer erzeugt werden. Die Kamera löst nämlich erst wieder aus, nachdem der Timer die laufende Belichtung beendet hat und die nächste startet.

Bei bestehender Bluetooth-Verbindung wird der aktuelle Belichtungsstatus kontinuierlich angezeigt (sofern nicht mit 'U'pdates abgeschaltet). Dabei überwacht der Intervalltimer auch, ob für den Shutter-Emulator tatsächlich Auslösungen erfolgen und zeigt eine Warnung an, wenn nicht. So kann man z.B. erkennen, wenn  die Speicherkarte voll ist.
Im "Schnelle Auslösungen" Modus wird jede Sekunde ein Punkt ausgegeben, der durch ein ' ersetzt wird, wenn der Shutter-Emulator gerade ausgelöst hat.
Kühler-Steuerung, PID-Regler

Befehle zur Kühler-Steuerung bestehen aus einer Zahl, gefolgt von einem 'K'. So schaltet '0K' den Kühler ab, während ihn '1K' bis '5K' mit unterschiedlicher Stärke einschalten. Wenn sich eine akzeptable Tempertur, die sich mit '?' anzeigen lässt, eingestellt hat, kann man mit '8K' den PID-Regler anweisen, die Temperatur zu halten.
Mit 'K' allein wird eine Übersicht der 'K'-Befehle angezeigt.
Mit weiteren 'K'-Befehlen, kann man auch eine Zieltempertur auswählen oder die Fenster-Heizung regeln.

Nach dem Einschalten ist der Kühler zunächst inaktiv, sofern nicht eine abweichende Einstellung im EEPROM gespeichert wurde (s.o.). 
Nach Erreichen des eingestellten Serien-Limits wird die Heizung abgeschaltet, um zu vermeiden, dass der Sensor wegen der geringeren Wärmeproduktion der abgeschalteten Kamera beschlägt.

Mit den 'C'-Befehlen lässt sich der PID-Regler rekonfigurieren, aber das sollte eigentlich nicht nötig sein.


Kühlwirkung

Nach dem Einschalten der Kühlung sinkt die gemessene Temperatur sehr schnell, was zuerst mal noch nicht viel aussagt, weil der Temperatursensor direkt auf dem Kühlblech liegt. Aber auch die EXIF-Temperatur der dabei aufgenommenen Bilder fängt schon nach 2 bis 3 Minuten an, stark abzusinken und kommt nach 10 bis 20 Minuten wieder in den Bereich des Tempertursensors. Das ist wesentlich effektiver als ich erwartet hatte. Anfangs hatte ich deshalb die Befürchtung, dass die EXIF-Temperatur nicht vom Sensor selbst stammt, sondern von einem NTC auf der Sensorplatine, der bei deren Kühlung die Sensortemperatur nicht mehr gut wiedergibt.
Die Messung des Dunkelstromrauschens des Sensors zeigt jedoch, dass der Sensor tatsächlich recht deutlich gekühlt wird, was gut zur EXIF-Temperatur passt. Was mich dann vollends überzeugt hat, war das starke Beschlagen des gekühlten Sensors bei seinem ersten Außeneinsatz. Ausgehend von meiner gekühlten 6D, die einen ähnlichen integrierten Kühler (mit Kühlfinger direkt unter dem Sensor) hat, hatte ich erwartet bis maximal ein bis zwei Grad unter die Umgebungstemperatur zu kommen und hatte deshalb zunächst keine Fensterheizung eingebaut, die ich jetzt erst mal nachrüsten musste. Damit habe ich dann die folgenden Werte am Temperatursensor gemessen, die etwa ein halbes Grad unter der EXIF-Temperatur liegen:

Kühlungseinstellung 1K 2K 3K 4K 5K 0K
PWM-Signal (D5) 255 164 71 42 0 -
Peltierspannung 0.95V 2V 3.1V 3.4V 4V -
Peltierleistung 0.25W 1W 2.5W 3W 3.8W -
Fensterheizung 35mW 66mW 400mW 460mW 620mW -
ΔT (mit 60s Frames)    -1.4°C  -3.0°C  -4.8°C  -4.6°C  -5.0°C  +6.4°C

Man kommt also bis etwa 5°C unter Umgebungstempertur, ohne Fensterheizung noch ein Grad tiefer. Die  fast volle Kühlwirkung hat man schon mit 2.5 Watt Peltier-Leistung, aber selbst mit einem viertel Watt bleibt die Sensortemperatur deutlich unter der Umgebungstempertur. Um tiefer zu kühlen müsste man den Lüfter hochdrehen oder den Kühlkörper vergrößern. Oder zwei Peltiers hintereinander schalten, aber dafür müsste man erst mehr Platz schaffen. Und der Gewinn wäre auch eher gering. Selbst bei 20°C Außentemperatur erreicht das Dunkelstromrauschen des dann 15°C warmen Sensors das Ausleserauschen erst bei einer Belichtungszeit von etwa 15 Minuten.
Bei abgeschalteter Kühlung erwärmt sich der Sensor um ca. 7 Grad bzw., wenn gar kein Kühler eingebaut ist, um etwa 11 Grad. Die gesamte Temperaturabsenkung gegenüber dem Originalzustand liegt dann also bei 16 Grad.

Hier ist eine Messreihe zur Kühlung, bei der die Kamera kontinuierlich 60s-Frames aufnimmt, zunächst ungekühlt, ab Frame 113 dann mit 2.5W Kühlleistung (bei aktiver Fensterheizung und Display aus):


Zunächst ist zu sehen, dass die Temperaturwerte vom NTC und aus den EXIF-Daten der Bilder über eine Stunde fast parallel ansteigen und beim Einschalten der Kühlung steil abfallen. Dabei folgt der Sensor dem NTC mit ca. 3 bis 5 Minuten Verzögerung. Dass hier auch wirklich die Sensortemperatur fällt, erkennt man daran, dass die aus den Bildern berechnete Standardabweichung der Temperaturkurve folgt.
Die Standardabweichung für Bias-Frames, also das Rauschen ohne Dunkelstrombeitrag beträgt für diese Kamera 13.5 ADU (bei ISO 1600), liegt für eine genaue Auswertung also zu nahe bei der Messkurve. Ein Vergleich mit 20-Minuten Aufnahmen, bei denen der Dunkelstrom einen höheren Anteil am Rauschen hat, ergibt, dass die EXIF-Temperatur die Sensortemperatur gut wiedergibt. Die im Diagramm auch noch eingezeichnete Größe der Bilddateien folgt übrigens auch sehr schön der Standardabweichung.

Aus der Verteilung der Standardabweichung über die Sensorfläche kann man erkennen, dass die Sensortemperatur ziemlich homogen ist. Selbst unter dem Kabelanschluss des Sensors, wo das Kühlblech nicht hinreicht, lässt sich keine höhere Temperatur feststellen. Offenbar hat der Sensor wirklich eine sehr gute Wärmeleitung:


Es gibt zwar einen gewissen Unterschied zwischen den Bereichen in der linken und rechten Sensorhälfte, aber die Mitte hat keine höhere Standardabweichung als die (im Bild) linke Sensorhälfte.


Fazit

Die flächig auf eine Platine gelöteten Sensoren der neueren Canon Cameras lassen sich viel besser kühlen als ich erwartet hatte, etwas besser sogar als die ältere Generation durch einen Kühlfinger im Spalt zwischen Sensor und Platine. 
Dazu kommt, dass es deutlich einfacher geht, man nicht einmal den Sensor dazu ausbauen muss. Schon mit sehr geringer Kühlleistung lässt sich die Sensortemperatur auf Umgebungstemperatur oder etwas darunter halten, so dass sich der Dunkelstrom auch im Sommer in akzeptablen Grenzen hält - und das alles mit einem voll integrierten Kühler ohne extra Verkabelung.
Und natürlich kann die integrierte Regelung die Temperatur konstant halten, so dass die Korrektur mit Darks viel leichter und zuverlässiger klappt.


Update: Feuchtesensor gegen Beschlagen des Sensors

In einigen Nächten im Oktober hatte ich den Fall, dass der Sensor selbst mit der kleinsten Kühlleistung im Lauf der Nacht beschlug. Deshalb habe ich die Schaltung für die Fensterheizung leicht modifiziert, so dass sie bei schwacher Kühlung stärker heizt. Und zur besseren Kontrolle der Beheizung hat der Arduino noch einen Temperatur- und Feuchtesensor vom Typ AHT20 bekommen, ein kleines Platinchen für einen Euro, das leicht unterzubringen ist und über die I2C Schnittstelle des Arduino angesprochen wird.
Aus der Temperatur und der relativen Luftfeuchtigkeit lässt sich der Taupunkt berechnen und damit abschätzen, ab welcher Temperatur der Sensor beschlagen wird. Damit kann ich nun die Sensor-Heizung an die aktuelle Wettersituation anpassen und im Fall extremer Luftfeuchtigkeit die Kühlung zurückfahren. 

Im Zuge der Aufrüstung mit dem AHT20 habe ich auch noch die Belegung einiger Steuerleitungen geändert, so dass am Arduino die für die SPI Schnittstelle benötigten Pins D11, D12 und D13 sowie A0 wieder frei sind. Statt diesen werden jetzt drei der Analog-Pins, die sich auch digital nutzen lassen, verwendet.  Insgesamt sieht die Schaltung jetzt so aus:

Die Aktivierung der beiden Mini360 übernimmt jetzt A3, während A1 und A2 die Verbindung mit dem Bluetooth-Modul herstellen. A4 und A5 bilden die I2C Schnittstelle, an der das AHT20 hängt.
Einer der vier 5.1Ω Widerstände im Spiegelkasten wurde durch zwei Dioden ersetzt, die die jeweils höhere der Peltier- und der Lüfter-Spannung auf die Heizwiderstände leiten. Da die Dioden mitheizen, sind sie ebenfalls im Spiegelkasten untergebracht.

Die Temperaturregelung des Arduino kann entweder direkt die eingestellte Kühlleistung reduzieren, wenn die Sensortemperatur den Taupunkt unterschreitet oder bei laufender PID-Regelung die Zieltemperatur anheben, wenn der Taupunkt im Lauf der Nacht steigt. Alternativ oder parallel dazu kann sie auch die Heizleistung anpassen.

Über die jetzt frei gewordene SPI Schnittstelle des Arduino (Pins D11, D12 und D13) plane ich noch, ein kleines Speichermodul zu integrieren, mit dem dann ein Software-Update über Bluetooth möglich sein wird, so dass ich die Kamera nicht mehr öffnen muss, wenn ich etwas anpassen möchte.


Update: Software-Update für den Arduino über Bluetooth

Ein Software-Update über Bluetooth hat den großen Vorteil, dass man die Kamera nicht öffnen muss, um Verbesserungen oder Erweiterungen am Steuerprogramm vorzunehmen. Voraussetzung dafür ist zunächst ein kleines externes Speichermodul vom Typ SPI-Flash, z.B. ein W25Q32. Dabei reicht schon eine kleine Größe (32 MBit bzw. 4MB) völlig aus. 
Weiterhin muss der Bootloader des Arduino-Boards durch eine passende Variante des fortschrittlicheren 'urboot' Bootloader ersetzt werden. Wenn dies korrekt eingerichtet ist, kann mit den 'F'-Funktionen der Arduino-Steuerung ein neuer Sketch in den externen Flash-Speicher geladen werden, der dann vom Bootloader nach einem Reset installiert wird. 
Ein neuer Sketch wird dazu auf die übliche Weise kompiliert und die dabei erzeugte Datei "ShutterEmuEOS.ino.hex" wird auf ein Gerät übertragen, das sich mit dem Arduino verbinden lässt. Ich schicke sie dazu per Email an das Smartphone, von wo sie mit dem Serial Bluetooth Terminal per Bluetooth an den Arduino hochgeladen wird.  
Es sollte aber auch gehen, sie aus einer Windows Kommandozeile mit dem copy-Befehl an eine (virtuelle) serielle Schnittstelle (z.B. COM9:), an die der Arduino als Bluetooth-Gerät gebunden ist, zu senden. (Siehe Windows Einstellungen -> Geräte -> Bluetooth- und andere Geräte -> Weitere Bluetooth-Optionen -> Reiter COM-Anschlüsse). Parallel dazu braucht man aber noch eine Terminalverbindung, um den Upload zu starten und abzuschließen.
Die größte Herausforderung ist dabei die Installation des Bootloaders. Dazu verwende ich Urboot zusammen mit MiniCore. MiniCore ist ein schlanker Core für den Arduino, der zusammen mit Urboot fast 2 KB Programmspeicher für eigene Sketche frei macht und Urboot direkt installieren kann. Leider jedoch nicht die spezielle "dual-boot" Variante von Urboot, die muss man von Hand nachinstallieren. 
Das wird mit dem Programm avrdude erledigt, dessen Aufruf sehr kompliziert aussieht. Am einfachsten ist es, die Aufrufe, die MiniCore in der Arduino-IDE (mit der Einstellung 'Show verbose output during upload') absetzt, zu kopieren und anzupassen. 
Der erste Teil davon sieht so oder so ähnlich aus: 

   "C:/Users/XYZ/AppData/Local/Arduino15/packages/MiniCore/tools/avrdude/7.2-arduino.1/bin/avrdude"
        "-CC:/Users/XYZ/AppData/Local/Arduino15/packages/MiniCore/hardware/avr/3.0.1/avrdude.conf"

und braucht nie verändert werden. Dann folgen einige Einstellungen, die schon angepasst sind: 

     -v -patmega328p -cstk500v1 -PCOM10 -b19200 

und zum Schluss kommen die Anweisungen, was zu tun ist, z.B. 

     -D "-Uflash:w:C:/Users/XYZ/AppData/Local/arduino/sketches/..../ShutterEmuEOS.ino.hex:i"  

um einfach einen Sketch zu laden, oder 

     "-Uflash:w:urboot_m328p_1s_autobaud_uart0_rxd0_txd1_led+d5_csb0_dual_ee_ce_hw.hex:i"
      -Ulock:w:0xff:m

um einen bestimmten Bootloader zu installieren. 
 Leider installiert MiniCore einen sogenannten 'Vector'-Bootloader. Das hat den Nachteil, dass ein Sketch, bevor er geladen wird, erst noch modifiziert werden muss, was normalerweise avrdude erledigt. 
Beim Laden über Bluetooth wird avrdude aber nicht verwendet, so dass man den Sketch selbst patchen müsste. Deshalb ist es besser, einen hardwareunterstützten Bootloader zu verwenden, dazu unten mehr. 
Falls (durch MiniCore) ein Vector-Bootloader installiert wurde, muss man im Prozessor des Arduino die Hardwareunterstützung des Bootloaders wieder aktivieren. Das geschieht durch Brennen einer sogenannten 'fuse' mit folgender Anweisung für avrdude: 

    -e -Ulock:w:0xff:m -Uefuse:w:0b11111101:m -Uhfuse:w:0xd6:m -Ulfuse:w:0b11110111:m 

Die verschiedenen fuses werden damit in den Zustand gebracht, wie sie MiniCore in der Standardeinstellung (für BOD 2.7V) und minimale Bootloader-Größe setzt, mit dem Unterschied, dass hfuse den Wert 0xd6 statt 0xd7 erhält, was die Hardwareunterstützung des Bootloaders aktiviert. 

Auswahl des Bootloaders
 
Im urboot.hex-Repository gibt es eine riesige Auswahl an vorcompilierten Bootloadern, von denen für einen Arduino Nano oder Pro-Mini aus diesem Bereich diejenigen infrage kommen, die auf '_dual' enden, zum Beispiel "led+d5_csb0_dual". 
Die Kennungen vor dem '_dual’ beschreiben, welche Pins zum Ansteuern einer LED ("led+" oder "led-") und als Chip-Select (CS) des SPI-Flashs ("cs")verwendet werden.  
Dabei ist b0 = PB0 = D8, b1 = PB1 = D9 und d5 = PD5 = D5. Die LED interessiert dabei nicht wirklich, weil sie ausgebaut wird. (Auf dem Pro-Mini oder Nano Board hängt sie an D13 und könnte deshalb eh nicht verwendet werden, weil D13 das SPI-Flash ansteuert.)

Allerdings sollte der LED-Pin nicht mit der Kamera in Konflikt kommen, wenn er vom Bootloader als Ausgang konfiguriert wird. In meiner 2000D verwende ich D5 als PWM-Ausgang. Damit ist die Bootloader-Auswahl mit "led+d5" (oder "led-d5") kompatibel. Für den Chip-Select bleibt dann b0, also D8, übrig. Unter dem "led+d5_csb0_dual" Link finden sich dann sechs Varianten, von denen eine mit "_hw" endet: 

    urboot_m328p_1s_autobaud_uart0_rxd0_txd1_led+d5_csb0_dual_ee_ce_hw.hex 

Wenn D8 schon anderweitig verwendet wird, gibt es als Alternative mit freier Pin-Wahl die template_dual Variante, in der LED und CS noch ohne Funktion sind. Hier muss man an den passenden Stellen geeignete Opcodes einfügen (patchen), damit der Bootloader 'dual' wird. Die geeignete Variante ist hier 

    urboot_m328p_1s_autobaud_uart0_rxd0_txd1_template_dual_ee_ce_hw.hex

Die passenden Stellen für CS-Anweisungen erkennt man an den Opcodes 0x2C55 bis 0x2C99 der Befehle
 "mov r5,r5" bis "mov r9,r9", die offensichtlich nicht viel tun. Sie werden ersetzt durch die passenden Portmanipulationsbefehle. Hier das Beispiel für PC0 alias Arduino A0 von Stefan Rueger

Template opcode     Replacement               Comment      
mov r5,r5 sbi <port-C>, 0 Set bit 0 of PORT C (release CS)      
mov r6,r6 cbi <port-C>, 0 Clear bit 0 of PORT C (assert CS)      
mov r7,r7 sbi <ddr-C>, 0 Set bit 0 of DDR C (make CS output)      
mov r8,r8 out <port-C>, r1 Reset all PORT C bits: register R1 contains 0      
mov r9,r9 out <ddr-C>, r1 Reset all DDR C bits      

Ohne geeignete Tools muss man die Opcodes selbst zusammenstellen. Es sind immer 2-Byte Codes, wobei das erste Byte im binären Programm an zweiter Stelle steht. 
Das zweite Byte besteht aus den Kennungen der Operanden (Bit-, Port- oder Register-Index): 

sbi: 0x9A + 5 Bits für das Port-Register + 3 Bits für das Bit; 
cbi: 0x98 + 5 Bits für das Port-Register + 3 Bits für das Bit; 
out: 0xB8 + 4 Bits für das Quellregister + 4 Bits für den I/O-Port. 

Die Adressen von PORT B/C/D sind 5/8/11 und die der DDR B/C/D sind 4/7/10.
(Weitere Informationen zum Patchen von Bootloader Templates.)
 
Da es sich bei Hex-Dateien um reinen Text handelt, kann das Suchen und Ersetzen der Opcodes mit einem Texteditor erledigt werden. Das sind die Ersetzungen, die ich auf das dual_template anwenden musste, um
A0 (= PC0) als Chip-Select zu verwenden:

Assembler-
Anweisung
Opcode   Hex           ersetzt
 durch:       
                 
mov r5,r5 0x2C55 552C
sbi 8, 0 0x9A40 409A  
mov r6,r6 0x2C66 662C cbi 8, 0 0x9840 4098  
mov r7,r7 0x2C77 772C sbi 7, 0 0x9A38 389A  
mov r8,r8 0x2C88 882C out 8, r1 0xB818 18B8  
mov r9,r9 0x2C99 992C out 7, r1 0xB817 17B8  

Die 'Hex'-Spalte zeigt, wie die Opcodes in einem Texteditor angezeigt werden. Nach dem Einsetzen habe ich die Bootloader-Datei als 

    urboot_m328p_1s_autobaud_uart0_rxd0_txd1_no-led_csc0_patched_dual_ee_ce_hw.hex 

 gespeichert. Das Ende jeder Zeile einer Hex-Datei bildet ein Prüfsummenbyte, das nun für alle gepatchten Zeilen ungültig ist. Um zu vermeiden, dass avrdude dafür Fehler meldet und abbricht, muss beim Aufruf an den Namen des hex Files als Format ":I" statt ":i" angehängt werden. 
 
Ein großes Danke geht an die Autoren von Urboot und MiniCore für ihre exzellente Software und wertvolle Hilfe! 
 
Ausführen eines Updates 
 
Für alles, was mit dem Flash-Speicher zu tun hat, sind die 'F'-Befehle zuständig: 

"0F" zeigt das erste Stück des externen Flashs an. 

"1F" startet das Übernehmen einer Datei, die dann im Menü des Serial Bluetooth Terminal mit "Upload File" ausgewählt und gesendet wird. Wegen der geringen Übertragungsrate von 9600 Baud dauert das ca. 2 bis 3 Minuten. Höhere Baudraten funktionieren möglicherweise nicht, wenn das Schreiben in den Flash zu langsam ist. 
Zuerst meldet das Serial Bluetooth Terminal, dass der Upload abgeschlossen ist und einige Zeit später sollte die Kamera dann bestätigen, dass der Sketch geladen und geprüft ist. Erst jetzt dürfen wieder Befehle an die Kamera geschickt werden, da die Kamera Eingaben und Upload nicht unterscheiden kann!

"2F" aktiviert den im externen Flash liegenden Upload und veranlasst einen Reset des Arduino, so dass der Bootloader den Upload als neues Steuerprogramm installieren kann.
Achtung: Neue Sketche sollten die Upload-Funktionen beinhalten, sonst geht die Update-Fähigkeit verloren!

Hardware

Zum Schluss noch der Anschlussplan des Flash-Moduls. Die letzten freien Pins neben D11-D13 waren A6 und und A7, die sich beim ATMega328P aber leider nicht digital nutzen lassen. Deshalb musste ich den NTC von A0 auf A6 umlöten und A0 als Chipselect für das Flash verwenden. Um sicher zu gehen, habe ich auch noch die LED des Flash-Moduls entfernt.




Zuletzt aktualisiert am 22.12.2024