Atmel programlayıcınız yoksa ve bootloader yüklü kontrolcü satın almadıysanız, aşağıdaki yöntemle ATMega168 ve AtMega328 kontrolcülerine bootloader kodunu atabilirsiniz. Tabi bunun için elinizde en azından bir Arduino kartı olması gerekiyor.
UNO kartını bilgisayara takıyoruz ve "Examples" klasöründeki ArduinoISP kodunu IDE'ye yüklüyoruz.
Arduino 1.0 sürümünü kullanıyorsanız bu kod içinde bir değişiklik yapmak gerekiyor. Bunun için; kod içinde heartbeat fonksiyonunu aratın ve fonksiyon içindeki delay(40) değerini, delay(20)olarak değiştirin. Bu kodu karta yükleyin ve alttaki devrelerden birini kurun.
Ben ATMega328P-PU için sağdaki devreyi kurdum ancak RESET ucundaki 10k direnci kullanmadım. Bağlantılar aşağıdaki şekilde;
UNO D10 -> ATmega RESET (1) UNO D11 -> ATmega SCK (19)
UNO D12 -> ATmega MISO (18)
UNO D13 -> ATmega MOSI (17)
UNO 5V -> ATmega VCC (7,20)
UNO GND -> ATmega GND (8,22)
Bunların dışında; isterseniz UNO'ya status ledlerini de bağlayabilirsiniz.
D9: Heartbeat - shows the programmer is running D8: Error - Lights up if something goes wrong (use red if that makes sense) D7: Programming - In communication with the slave
Seri port seçimini de yaptıktan sonra, Programlayıcı olarak "Arduino as ISP" seçiyorsunuz ve <Burn Bootloader> 'ı seçiyorsunuz. Birkaç saniyede, bootloader kodu yükleniyor.
Uno'nun yeni sürümlerinde deneme şansım olmadı. Yüklemede sorun yaşarsanız, ArduinoISP kodunu attıktan sonra, Uno'nun reset ucunu 10uF kondansatör ile şaseye bağlayarak deneyebilirsiniz. Ardunio sayfasında bu konuyla ilgili detaylı bilgi bulabilirsiniz. link
Daha önce şemasını verdiğim 220V AC dimmer devresine, DMX512 protokolü ile kontrol edilebilen ve 4 adet lambayı sürebilen bir kart tasarlamak için başlamıştım. Ledler yaygın olarak kullanılmaya başlandığından beri 220V AC dimmer ihtiyacı pek kalmadı ama bazı uygulamalar için hala gerekebiliyor. Devre, bir mikrokontrolcü, bir sıfır geçiş dedektörü ve dört adet AC sürücü katından oluşuyor. Mikrokontrolcü olarak 16F73 kullandım, 28 pin herhangi bir PIC de kullanılabilir.
sıfır geçiş dedektörü
AC altında triac sürme konusundan daha önce bahsetmiştim, bununla ilgili 220V AC DIMMER konusuna bakabilirsiniz.
Burada farklı olarak, 4 kanal için parlaklık ayarı, başka bir cihaz (ışık masası gibi) veya PC yazılımı üzerinden yapılabiliyor. Bunun için DMX-512 protokolü kullanılıyor. Her lamba, 0-255 kademe aralığında sürülebiliyor.
DMX-512 Sender
Burada kullandığım DMX sender kartı, USB'den aldığı dataları, hatta bağlı dimmer kartlarına gönderiyor. Bir hat üzerinde 512'ye kadar adres tanımlanabiliyor. Dimmer devresinde kullanılacak yük akımına göre triak seçimi yapılarak, 5000KW güçlere kadar flaman ampullerin dimlenmesi mümkün. bunun için BTA41 kullanabilirsiniz.
Elm ve OBD konusunda bahsetmiştim, araç bilgilerini okumanın başka bir yolu da, araç içinde dolaşan farklı canbus hatlarına girmektir. Bus sayısı, hızları, ve içerdiği bilgiler farklı olabiliyor ancak genelde tüm araçlarda CANBUS protokolü kullanılıyor.
Farklı hatlar, farklı bilgiler topluyor ve bu hatlar aynı zamanda, gateway ünitesi üzerinden birbirleri ile de haberleşiyor.
gateway
OBD portundan alınan bilginin fazlasını, diğer hatlardan almak mümkün. VAG araçlarına ait canbus diagramı aşağıdaki gibi.
Burada şeması verilen devre, bir CAN kontrolcü ve mikrokontrolcüden oluşuyor. CAN kontrolcü olarak Mikrochip firmasının MCP2515 entegresi, mikrokontrolcü olarak da, Arduino nano kartı kullandım.
Arduino'nun (veya ATMEGA'nın) MCP2515 bağlantısı için SPI ve INT pinleri kullanılıyor. MCP2515'in RST ucu, arduino kartının reset ucuna bağlanacak. (hardware reset için) Devre 3 modda çalışabiliyor.
mod-1 Sniffer
mod-2 OBD2 Reader
mod-3 Sender
Mod seçimi D6 ve D7 pinlerine bağlı switch ile yapılıyor. Şemada göstermedim ama D6 ve D7 pinleri 10k direnç ile şaseye çekilmeli. anahtar "B" konumunda ise; devre, periyodik olarak, bazı OBD komutlarını gönderiyor ve bu komutların yanıtlarını alıyor. Bu, bağlantı testi için kullanılabilir. Aynı kodları veya daha fazlasını "C" konumunda da gönderebilirsiniz. OBD için gönderilen komutlar: ENGINE_COOLANT_TEMP (0x05) ENGINE_RPM (0x0C) VEHICLE_SPEED (0x0D) INTAKE_AIR_TEMP (0x0F) CONT_MODULE_VOLT (0x42) AMBIENT_AIR_TEMP (0x46) CATALYST_TEMP_B1S1 (0x3C)
Switch "A" konumunda ise, canbus hattındaki data trafiği dinleniyor. Filtre veya mask tanımlanmamış ise, hat üzerindeki tüm datalar izlenebilir.
Switch "C" konumunda ise, yazılan komutu can hattına gönderiyor ve yanıtını bekliyor.
Hattan alınan tüm datalar, tanımlanmış bir formatta, arduino'nun seri çıkışına gönderiliyor.
PC arayüzü bu dataları alıp tabloya yazıyor. Bu arayüz üzerinden, hat hızı, maske ve filtre tanımlamak ve komut göndermek de mümkün.
Başta da dediğim gibi, araç içindeki tüm can hatları gateway'de birleşiyor ve gateway
bu hatlar üzerinde geçiş sağlıyor. OBD soketi üzerinden de bu hatlara geçiş olduğunu düşünüyorum ama bu konuda fazla deneme imkanım olmadı. Bazı markalarda, OBD soketinde, diagnostik hattının haricindeki hatlar da bulunabiliyor. Şayet yoksa, yapılması gereken, hattın geçtiği bir noktadan, paralel uç almak.
Örneğin, vw grubunda, radyo soketindeki can hattına bağlanıp data akışını dinlemek ve
buradan bazı işlemler yapmak mümkün. (direksiyondaki butonları okumak, radyo veya yol bilgisayarı ekranına yazı yazmak gibi) Diğer bilgiler için başka hatlara bağlanmak gerekiyor.
Gateway soketinde araya girip 3 CAN hattından da uç almak en iyi çözüm bence.
Yukarıdaki şema için hex dosyasını ve PC yazılımını aşağıdaki linklerden indirebilirsiniz.
Bir ihtiyaçtan dolayı akkor lambalar için dimmer yapmam gerekiyordu. Finalde 4-8 kanal ve PC/DMX kontrollü bir kart olacak. AC sinyalde triac tetikleme ile ilgili dökümanlar, nette mevcut, o nedenle fazla detaya girmiyorum. Kısaca bahsetmek gerekirse, Çalışma mantığı şöyle;
Şebeke sıfır geçiş noktasını bul
istenilen gecikme kadar bekle
triac'ı tetikle
her alternansta bu işi tekrarla. AC gerilim altında, triac'ı 1 kez tetiklediğinizde, bir dahaki sıfır geçişine kadar iletimde kalır. 2 geçiş noktası arasında (50Hz şebekede 10mS) ne kadar önce tetiklerseniz lamba, o kadar parlak yanar. Diyelimki lambayı %50 parlaklıkta yakmak istiyoruz. Sıfır geçişini tespit ettik, timer'ı sıfırladık, 5mS bekledik ve triac'ı tetikledik. sonraki 5mS boyunca (bir dahaki sıfır geçişine kadar) triac iletimde kalacaktır. Sıfır geçişini nasıl buluruz? Bunun da benzer bir çok yöntemi var. Benim kullandığım devre aşağıdaki gibi.
Bu devrenin çıkışını PIC'in RB0 pinine giriyoruz. Gerekli interrupt konfigürasyonu yapıldığında, her sıfır geçişinde bir kesmemiz oluyor. Bundan sonrası, gecikme zamanını saymak ve çıkışı tetiklemek. Triac sürücü devresi de aşağıdaki gibi.
ben, BT138 triac ve sürmek için de MOC30xx serisi optodiac kulllandım. PIC ile doğrudan triac sürmek de mümkün. Test devresinde snubber kullanmadım. İsterseniz, triac'ın A1-A2 uçları arasına; 39ohm ve 100nf'ı seri bağlayabilirsiniz.
PIC 16F628A xtal: 20MHz
RA0 portu Triac sürücü devresine bağlanacak, RB0 girişi de sıfır geçiş dedektörünün çıkışına. RA1'de led var, bu led, sıfır geçişi algılanıyorsa, periyodik olarak yanıp söner. Triac, 255 kademede sürülebilir. Kademe (parlaklık) ayarı için RA2 ve RA3 portlarına bağlı butonları kullanabilirsiniz. Aşağıda, devrenin giriş ve çıkış sinyallerini görebilirsiniz. Kırmızı sinyal şebeke voltajıdır. Bu sinyalin sıfır geçişlerinde üretilen sinyal de yeşil ile gösterilmiştir. (zero dedektör çıkışı) Sarı ile gösterilen sinyal de yarı güçte sürülen triac'ın tetikleme sinyalidir.
Devreler test için yapılmıştır. Kullanılan bazı dirençler, seçilecek komponentlere göre yeniden hesaplanmalıdır. Aşağıdaki linkten hex dosyasını indirebilirsiniz. F628_Hex dosyası
Mikrochip firmasının USB destekli çiplerini uzun zamandır kullanıyorum. Özellikle HID protokolü ile kullanıldığında, sürücü gerektirmeden, hızlı bir haberleşme sağlıyor. bunun için yazılması gereken kodu da bazı programlar sizin için üretiyor. Size de gelen dataları işlemek ve göndermek kalıyor. Bu iş için benim kullandığım yazılım, Mecanique firmasının EasyHID yazılımı. Bu program, hem PIC için hem de PC tarafı için örnek kod üretiyor.
PIC tarafında üretilen kod için söylenecek fazla birşey yok. Yapmanız gereken; ProductID, VendorID ve data uzunluklarını belirlemek ve kodu oluşturmak. Burada, easyHID tarafından oluşturulan, cUSBInterface ve cUSBInterfaceTypes dosyalarının ve mcHID.dll dosyasının, yazdığınız programla aynı klasörde olması gerekiyor. Uygulama derlendikten sonra, dll dosyasının olması yeterli.
aşağıda, Delphi için oluşturulmuş, 8 byte data alıp gönderen bir kod örneği var. Sayfanın sonunda da, indirme linkini bulabilirsiniz.
HID kullanan diğer cihazları da, hazırladığınız yazılımla izleyebilirsiniz. Aşağıdaki uygulama, UPS bilgilerini, USB portundan okumakta ve ekrana yazmaktadır.
UPS'in Vendor ve product ID'lerini ve cihazdan gelen dataları, HID terminal yazılımı ile görebilirsiniz. Bu bilgileri aldıktan sonra, yapmanız gereken; Vendor ve product ID'lerini, kodun başında tanımlamak ve USBEvent fonksiyonunda gelen bilgileri alıp işlemek.
----------------------------------------Örnek program kodu--------------------------------------------------
unit FormMain; interface uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls, OleCtrls, DBOleCtl, ComCtrls, VrControls, VrSlider; const // input and out buffer size constants... BufferInSize = 8; BufferOutSize = 8; type // input and output buffers... TBufferIn = array[0..BufferInSize] of byte; TBufferOut = array[0..BufferOutSize] of byte; // main form TForm1 = class(TForm) LabelProductName: TStaticText; LabelVendorName: TStaticText; LabelUsbStat: TStaticText; Button3: TButton; Button1: TButton; procedure FormCreate(Sender: TObject); procedure FormDestroy(Sender: TObject); procedure Button1Click(Sender: TObject); private FBufferIn:TBufferIn; FBufferOut:TBufferOut; function USBEvent(var Msg: TMessage): Boolean; function SendData: Boolean; public end; var Form1: TForm1; implementation uses cUSBInterface, cUSBInterfaceTypes, Math; const // vendor and product ID constants... VENDOR_ID = $1234; PRODUCT_ID = 0001; {$R *.DFM} { **************************************************************************** * Name : Create * * Purpose : Create the main form * **************************************************************************** } procedure TForm1.FormCreate(Sender: TObject); begin Application.HookMainWindow(USBEvent); Connect(Application.Handle); end; { **************************************************************************** * Name : Destroy * * Purpose : Free the main form * **************************************************************************** } procedure TForm1.FormDestroy(Sender: TObject); begin Application.UnHookMainWindow(USBEvent); end; { **************************************************************************** * Name : USBEvent * * Purpose : DLL message handler hook * **************************************************************************** } function TForm1.USBEvent(var Msg: TMessage): Boolean; var i: integer; DevHandle:cardinal; TextBuffer:array[0..255] of char; begin result := False; if Msg.Msg = WM_HID_EVENT then begin case Msg.WParam of // a HID device has been plugged in... NOTIFY_PLUGGED : begin // is it our HID device... DevHandle := Msg.LParam; // handle of HID device in this message if (GetVendorID(DevHandle) = VENDOR_ID) and (GetProductID(DevHandle) = PRODUCT_ID) then begin // add your code here, for example... GetProductName(DevHandle, TextBuffer, 256); LabelProductName.Caption := string(TextBuffer); GetVendorName(DevHandle, TextBuffer, 256); LabelVendorName.Caption := string(TextBuffer); LabelUsbStat.Caption := 'Plugged'; end; result := true; end; // a HID device has been device removed... NOTIFY_UNPLUGGED : begin // is it our HID device... DevHandle := Msg.LParam; // handle of HID device in this message if (GetVendorID(DevHandle) = VENDOR_ID) and (GetProductID(DevHandle) = PRODUCT_ID) then begin // add you code here LabelUsbStat.Caption := 'Unplugged'; end; result := true; end; // a HID device has been attached or removed. This event is fired after // either NotifyPlugged or NotifyUnplugged. NOTIFY_CHANGED : begin // get the handle of the device we are interested in // and set it's read notification flag to true... DevHandle := GetHandle(VENDOR_ID,PRODUCT_ID); SetReadNotify(DevHandle,true); result := true; end; // a HID device has sent some data.. NOTIFY_READ : begin DevHandle := Msg.LParam; // handle of HID device in this message if (GetVendorID(DevHandle) = VENDOR_ID) and (GetProductID(DevHandle) = PRODUCT_ID) then begin // read the data - remember that first byte is report ID... Read(DevHandle,@FBufferIn); // USB aygıttan gelen datalar burada işlenecek! end; result := true; end; end; end; end; //USB'ye data gönder function TForm1.SendData: Boolean; var r: boolean; begin r := WriteEx(VENDOR_ID, PRODUCT_ID, @FBufferOut); result := r; end; procedure TForm1.Button1Click(Sender: TObject); begin FBufferOut[0] := 0; FBufferOut[1] := 80; FBufferOut[2] := 70; FBufferOut[3] := 1; FBufferOut[4] := 0; FBufferOut[5] := 0; FBufferOut[6] := 0; FBufferOut[7] := 0; SendData(); //Fbuufferout'a yazılanlar gönderilir end; end.
Aracın bilgi ekranında gösterilmeyen bazı verileri izlemek, bazılarını da bir araya toplamak için küçük bir projeye başlamıştım. Henüz tam olarak bitmese de sona yaklaştım.
OBD2 soketi bazı markalarda farklılık gösterse de genelde resimdeki gibi 16 pinli bir soket.
Direksiyonun alt taraflarında veya orta konsolda oluyor, C4'lerde kül tablasının altındaydı örneğin. Astra'da vitesin arkasında, kapağın altına gizlenmişti. Seat'da sol altta, yine kapağın altında.
Donanım 2 bölümden oluşuyor.
1.bölüm, OBD portundan verileri alıp işleyen ve seri olarak displaye gönderen modül
2.bölüm, seri dataları işleyip ekrana basan 3.2" lik grafik ekran ve 320x240 çözünürlükte. Üzerinde kendi kontrolcüsü ve donanımsal seri portu var.
Alınan datalar:
Hız
Motor devri
Motor suyu sıcaklığı
Dış ortam sıcaklığı
Turbo hava girişi sıcaklığı
CAT sıcaklığı
AKÜ voltajı
Yakıt tüketimi ile ilgili bilgiler maalesef hazır olarak gelmiyor. Saatteki yakıt tüketim (lt/h) bilgisi alınabiliyor. ancak anlık tüketim, ortamalama tüketim vs. bilgilerin, bazı datalardan hesaplanarak çıkartılması gerekiyor. (Araç hızı, MAF sensör bilgisi gibi) Bunların dışında, OBD2'den gelen birçok data var, istenirse bunlar da işlenip kullanılabilir.
Verileri diagnostik portundan okumanın bir kaç yolu var.
ELM çipli bir devre (veya hazır modül) kullanabilirsiniz veya PIC/Ardunio gibi bir kontrolcü kullanabilirsiniz.
ELM kullanacaksanız, daha önceki konularda anlattığım gibi, AT komutlarını kullanabilirsiniz. Bağlantı kurulduktan sonra, mod ve pid değerlerini sorgulamanız yeterli.
ELM entegresini veya hazır devreleri kullanmak istemezseniz, diagnostik portla bağlantı kuracak bir arabirime ihtiyacınız olacak. Bunun için microchip firmasının 2515 ve 2551 canbus entegrelerini kullanabilirsiniz.
Canbus haberleşmesi, bir-iki satırla geçiştirilmeyecek kadar kapsamlı. O nedenle, bunu daha sonraya bırakıyorum. Ben, test için daha önce yaptığım CANBUS arabirimini kullandım.
Modül, OBD2/CANBUS hattına sürekli olarak yukarıdaki sorgu komutlarını gönderiyor ve aşağıdaki şekilde yanıt alıyor. Bu datalar belirli bir formata çevrilip, seri port üzerinden grafik LCD'ye gönderiliyor.
Display devresi, seri porttan gelen bilgileri işleyerek ekrana yazıyor. Arabirim ve GLCD 5V ile çalışıyor. Deneme için diagnostik portundaki 12V çıkışını kullandım. Kontak kapalı olduğunda bile bu uçta gerilim var. Test için kullanılabilir ancak kalıcı bir uygulama için başka bir yerden +12V almak daha mantıklı.
Yukarıda bahsettiğim grafik display, ELM327 modülünü AT komutları ile sorgulayacak ve gelen dataları işleyecek kapasitede. ELM modülü ile ek bir devre olmadan da okuma yapılabilir.
Burada karşıma çıkacak en büyük sorun, kullandığım ekrana, araç içinde uygun ve güzel bir yer bulmak. tabi bu her model araç için farklı çözümler gerektiriyor. Proje istediğim gibi gelişirse, ilerleyen zamanlarda, aracın kendi multimedya sistemini kullanmayı düşünüyorum. VAG araçlarında (vw, seat, audi, skoda) kullanılan RCD/RNS510 medya cihazının, geri görüş kamerası için bir video girişi bulunuyor. Bu giriş bir kontak girişiyle tetikleniyor.
Okunan ve işlenen dataları PAL formatında göndermek, -zahmetli olsa da- mümkün. Yine bu bilgiler, MFD üzerinde de izlenebilir.
Elimizde ELM çipli bir okuyucu olduğunu düşünelim. bununla, OBD portundan bilgi okumak istiyoruz. Elinizdeki bluetooth destekli bir modelse, öncelikle yapmanız gereken, bluetooth bağlantısını kurmak ve bağlantı için atanan sanal COM portun hangisi olduğunu belirlemek. Hyperterminal programını açın, modülü bağladığınız portu seçin. Baud hızını ayarlayın. (modüle göre değişmekle beraber genelde default olarak 38400 baud kullanıyorlar) Bağlantı kurulduğunda veya 'ATZ' komutu gönderdiğinizde aşağıdaki gibi bir yanıt gelmeli.
ATZ ELM 327 v1.X >
Bu yanıtı aldıysanız bağlantı kurulmuş demektir. ELM birden fazla protokolü destekliyor ancak ben burada, CAN protokolünü kullandığımızı varsayıyorum. Protokol ve hızı otomatik ayarlaması için aşağıdaki komutu gönderiyoruz. >ATSP 0 >OK Artık istediğiniz bilgileri sorgulayabilirsiniz. Sorgulama kodu 'Mode' ve 'PID' olmak üzere 2 bayttan oluşur. Mode 01, anlık dataları okumak için kullanılır. PID ise, okunacak sensör bilgisinin kodudur. örneğin; PID 0x05, 'Soğutma suyu sıcaklığı' değeridir.
'Soğutma suyu sıcaklığı' değerini okumak için aşağıdaki şekilde mode ve PID değerini gönderin. > 01 05 > 41 05 7B burada 7B sorgumuzun sonucudur, değeri desimale çevirip formülü uygularsak (bu komut için A-40) 123-40 = 83°C değerini buluruz. Araçların çoğu standard PID'leri desteklese de, her araçda tüm komutlar okunamayabiliyor. Aracınızın hangi PID'leri desteklediğini görmenin de kolay bir yolu var. PID 00, 20, 40 ,60 ve 80 aracın desteklediği PID numarasını geri döndürür. >01 00 sorgusunun yanıtı 4 bayttır ve bu 4 bayt, 0x01 ile 0x20 arasındaki 32 adet PID'in hangilerinin desteklendiğini gösterir. Örneğin: >01 00 >41 00 BE 1F A8 13
B E 1 F A 8 1 3
---- ---- ---- ---- ---- ---- ---- ----------
supported? 1011 1110 0001 1111 1010 1000 0001 0 0 1 1
PID num 1234 5678 9ABC DEF. .... .... .... 1D 1E 1F 20
Bu şekilde, aracınızın desteklediği komutları alabilir ve sorgulama yapabilirsiniz. Burada sadece anlık sensör bilgilerinin alınması konusunda bilgi vermeye çalıştım. Diğer modlarda arıza bilgilerini almak, arıza ışığını söndürmek, araca ait diğer bilgileri almak mümkün.
OBD2 ile ilgili örnek uygulama için <Araç Bilgi Ekranı> linkine bakabilirsiniz.