<?xml version='1.0' encoding='UTF-8'?><?xml-stylesheet href="http://www.blogger.com/styles/atom.css" type="text/css"?><feed xmlns='http://www.w3.org/2005/Atom' xmlns:openSearch='http://a9.com/-/spec/opensearchrss/1.0/' xmlns:georss='http://www.georss.org/georss' xmlns:gd='http://schemas.google.com/g/2005' xmlns:thr='http://purl.org/syndication/thread/1.0'><id>tag:blogger.com,1999:blog-5727727405189705833</id><updated>2011-04-22T01:00:01.623+02:00</updated><category term='pci'/><category term='usb'/><category term='irq'/><category term='bios'/><category term='electronics'/><title type='text'>Çekirdek Notları</title><subtitle type='html'></subtitle><link rel='http://schemas.google.com/g/2005#feed' type='application/atom+xml' href='http://cekirdeknotlari.blogspot.com/feeds/posts/default'/><link rel='self' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default?max-results=100'/><link rel='alternate' type='text/html' href='http://cekirdeknotlari.blogspot.com/'/><link rel='hub' href='http://pubsubhubbub.appspot.com/'/><author><name>Faik Uygur</name><uri>http://www.blogger.com/profile/15466697634443584707</uri><email>noreply@blogger.com</email><gd:image rel='http://schemas.google.com/g/2005#thumbnail' width='16' height='16' src='http://img2.blogblog.com/img/b16-rounded.gif'/></author><generator version='7.00' uri='http://www.blogger.com'>Blogger</generator><openSearch:totalResults>5</openSearch:totalResults><openSearch:startIndex>1</openSearch:startIndex><openSearch:itemsPerPage>100</openSearch:itemsPerPage><entry><id>tag:blogger.com,1999:blog-5727727405189705833.post-5005599605066240704</id><published>2007-10-13T22:08:00.000+02:00</published><updated>2007-10-13T22:10:04.489+02:00</updated><title type='text'>Taşındım</title><content type='html'>Wordpress'e döndüm. Yeni &lt;a href="http://blog.faikuygur.com"&gt;adresim&lt;/a&gt;.&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/5727727405189705833-5005599605066240704?l=cekirdeknotlari.blogspot.com' alt='' /&gt;&lt;/div&gt;</content><link rel='replies' type='application/atom+xml' href='http://cekirdeknotlari.blogspot.com/feeds/5005599605066240704/comments/default' title='Post Comments'/><link rel='replies' type='text/html' href='http://www.blogger.com/comment.g?blogID=5727727405189705833&amp;postID=5005599605066240704' title='0 Comments'/><link rel='edit' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/5005599605066240704'/><link rel='self' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/5005599605066240704'/><link rel='alternate' type='text/html' href='http://cekirdeknotlari.blogspot.com/2007/10/tandm.html' title='Taşındım'/><author><name>Faik Uygur</name><uri>http://www.blogger.com/profile/15466697634443584707</uri><email>noreply@blogger.com</email><gd:image rel='http://schemas.google.com/g/2005#thumbnail' width='16' height='16' src='http://img2.blogblog.com/img/b16-rounded.gif'/></author><thr:total>0</thr:total></entry><entry><id>tag:blogger.com,1999:blog-5727727405189705833.post-4130300564490306818</id><published>2007-02-25T18:34:00.000+02:00</published><updated>2008-12-09T17:24:33.126+02:00</updated><category scheme='http://www.blogger.com/atom/ns#' term='electronics'/><category scheme='http://www.blogger.com/atom/ns#' term='irq'/><title type='text'>Kenar (edge) ve seviye (level) tetiklemeli kesmeler üzerine</title><content type='html'>Çekirdek için donanım ile yazılım arasında bir arayüz diyoruz. Genelde donanım ile alakalı kısımları okurken, donanım hakkında bir fikir sahibi olmanız gerekiyor ki çekirdeğin nereye varmaya çalıştığını anlayabilesiniz. Bunun için de donanım speklerine ihtiyaç duyarsınız. Ama bazen bu spekler de yetersiz kalır, çünkü spekler size o donanımı nasıl kullanabileceğinizi anlatırlar. Fakat sizin ihtiyacınız olan, o an baktığınız sistemde o donanımın ne şekilde kullanılmış olduğu bilgisidir.&lt;br /&gt;&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;ISA&lt;/span&gt;'da &lt;span style="font-weight: bold;"&gt;edge-triggered&lt;/span&gt; kesmeler, &lt;span style="font-weight: bold;"&gt;PCI&lt;/span&gt;'da &lt;span style="font-weight: bold;"&gt;level-triggered&lt;/span&gt; kesmeler kullanıldığını ve bu yüzden PCI bus'larda kesme paylaşımı yapılabileceği ancak ISA'da kesme paylaşımı yapılamayacağını duymuş ama nedenini merak ediyorsanız, hatta ISA'da aynı zamanda kesme paylaşımı yapılabileceğini duymuş, hayda hani yapılamıyordu diyorsanız, ya da edge/level triggered ne ola ki diyorsanız okumaya devam edebilirsiniz.&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;a onblur="try {parent.deselectBloggerImageGracefully();} catch(e) {}" href="http://3.bp.blogspot.com/_VELeD1M_mEU/ReXoT_AsZHI/AAAAAAAAAAM/Wxshpu_9ms8/s1600-h/edge-level.png"&gt;&lt;img style="margin: 0px auto 10px; display: block; text-align: center; cursor: pointer;" src="http://3.bp.blogspot.com/_VELeD1M_mEU/ReXoT_AsZHI/AAAAAAAAAAM/Wxshpu_9ms8/s320/edge-level.png" alt="" id="BLOGGER_PHOTO_ID_5036687188313269362" border="0" /&gt;&lt;/a&gt;&lt;br /&gt;Dijital sistemlerde kesmeler sinyalin farklı zamanlarına bakılarak tetiklenebilir. İki çeşit tetikleme mekanizması bulunmaktadır: edge (kenar) ya da level (seviye) tetiklemeli. Kenar tetiklemeli kesmeler, sinyalin 1'den 0'a ya da 0'dan 1'e geçtiği anların tespiti ile; seviye tetiklemeli kesmeler, sinyal'in 0 ve ya 1 seviyesinde bulunduğu anlara bakılarak yapılır.&lt;br /&gt;&lt;br /&gt;Kenar (edge) ve seviye (level) tetiklemeli kesmeler ve sorunları üzerine konuşacağımız için Intel'in &lt;span style="font-weight: bold;"&gt;8259&lt;/span&gt; entegresi bu konu için ideal. 8259'un &lt;a href="http://bochs.sourceforge.net/techspec/intel-8259a-pic.pdf.gz"&gt;datasheet&lt;/a&gt;'ine buradan ulaşabilirsiniz.&lt;br /&gt;&lt;br /&gt;&lt;span style="font-size:100%;"&gt;&lt;span style="font-weight: bold;"&gt;Kesme Süreci&lt;/span&gt;&lt;/span&gt;&lt;br /&gt;&lt;br /&gt;x86 mimarisindeki bir işlemcide "genel" amaçlı kesmeler için istenildiğinde maskelenebilen INTR pin'i kullanılıyor. Pin dışarıdaki bir kesme kontrolörünün çıkışına bağlanıyor. Kontrolör tek bir 8259 entegresi olduğu durumda 8 farklı cihaz da bu kontrolörün girişine bağlanabiliyor.&lt;br /&gt;&lt;br /&gt;8259 kontrolörü içerisinde kesme isteklerini işleyen iki odacık olduğunu düşünebiliriz. İlk oda kabul odası (IRR), ikinci oda ise servis odası (ISR). Her hangi bir girdiden kesme geldiği anda IRR odasında beklemeye alınıyor ve özel bir süreç işlemeye başlıyor. Bu süreç şu şekilde işliyor:&lt;br /&gt;&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;1.&lt;/span&gt; Kontrolör girişlerinden bir pin (IR7-0) yüksek değere çekilir. Kesme geldi.&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;2.&lt;/span&gt; 8259 bunu farkeder ve işlemci'ye INT gönderir.&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;3.&lt;/span&gt; İşlemci, INTA pin'i ile kontorolöre ack sinyali gönderir.&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;4.&lt;/span&gt; IRR'deki yüksek öncelikli bit (0-7) ISR'ye alınır ve IRR'deki bit temizlenir.&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;5.&lt;/span&gt; İşlemci ilk INTA'nın ardından ikinci bir INTA gönderir. Bu sefer 8259, 8-bitlik veri hattına kesme vektör bilgisini yerleştirir. İşlemci bu vector bilgisini IDT (Interrupt Descriptor Table) ile birlikte kullanarak o kesme için işaret edilen adresteki kesme işleyicisini (interrupt handler) çalıştırır.&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;6.&lt;/span&gt; IRR, ilk INTA sonunda resetlenmişti, ikinci INTA sonunda da ISR resetlenmeli. Bu, kesme'nin görevini yerine getirdiğinin bir işareti: &lt;span style="font-weight: bold;"&gt;End-Of-Interrupt&lt;/span&gt; (&lt;span style="font-weight: bold;"&gt;EOI&lt;/span&gt;). 8259 Auto EOI modunda ise bu biti ikinci INTA sonrası kendisi temizliyor, eğer değilse kesme işleyicisinin ilgili ISR bit'i için 8259'a EOI komutu göndermesi gerekiyor.&lt;br /&gt;&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;Kenar Tetiklemeli Kesmeler (&lt;/span&gt;&lt;span style="font-weight: bold;"&gt;Edge-Triggered Interrupts)&lt;/span&gt;&lt;br /&gt;&lt;br /&gt;ISA veriyolu kenar tetiklemeli kesmeler kullanılmak üzere tasarlanmış. Anakart üzerinde bir yerlerde 8259 kesme kontrolörü kenar tetiklemeye duyarlı olarak programlanmış şekilde bekliyor. Kontrolör herhangi bir kesme girişinden sinyal'in düşük seviye'den yüksek seviyeye geçtiğini algıladığı anda işlemci'ye kesme gönderecek. Buna göre şu şekilde çalışan bir ISA cihazı düşünebiliriz. Cihaz normalde kesme hattını düşük seviyede, kesme göndereceği zaman da kesme hattını yüksek seviyede&lt;span style="font-weight: bold;"&gt; &lt;/span&gt;&lt;span style="font-weight: bold;"&gt;sürecek&lt;/span&gt;. Kesme gönderdikten sonra kesme işleyicisinden cihaz'ın kesme göndermesine sebep olan neyse bu durum çözümlenecek ve ya ileride halledilmek üzere bu bilgi not edilecek, sonra da kesme hattını tekrar düşük seviyeye çekmesi &lt;span style="font-weight: bold;"&gt;kesme işleyicisi (sürücü) tarafından&lt;/span&gt; sağlanacak. Bunun nasıl yapılacağı cihazdan cihaza değişiklik gösterebilir.&lt;br /&gt;&lt;br /&gt;Kenar tetiklemeli olmasına rağmen 8259'un tasarımı gereği ilk INTA geldikten sonra hattın aktif tutulması gerekiyor. Bunun sebebi tetiklemeyi yapan sinyal'in bir &lt;span style="font-weight: bold;"&gt;gürültü&lt;/span&gt; (noise) olabilme olasılığı. Kesme olmadığı zamanda düşük seviyede olan hatta oluşabilecek bir gürültü tetiklemeye yol açabilir. Hat o an aktif değil ise ikinci INTA'da işlemciye özel olarak 7 vektörü dönülüyor. Buraya gerçekten bir cihaz bağlı ise, kesme işleyicisi'nin kesme'nin gerçek olup olmadığını cihaz ile konuşarak kontrol etmesi lazım.&lt;br /&gt;&lt;br /&gt;Bu şekilde tasarlanmış kesme çıkışları ile yani hattı hem düşük, hem yüksek süren cihazların ikisi aynı hattı paylaşamaz.&lt;br /&gt;&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;Kenar Tetiklemeli Kesme Paylaşımı (Edge-Triggered Interrupt Sharing)&lt;/span&gt;&lt;br /&gt;&lt;br /&gt;Kenar tetiklemeli kesmeleri paylaştırabilmek için şöyle bir yöntem kullanılabiliyor: Öncelikle anakart üzerinde  ve ya kontrolörün içerisinde (bazı üretici modellerinde mevcut değil) her bir kesme hattı bir pull-up direnci ile yüksek seviyede tutuluyor. Paylaşım yapacak cihaz, kesme göndereceği zaman hattı bir süre düşük seviyede sürüyor, hattı sürmeyi bıraktığı anda hat otomatik olarak pull-up direnci sayesinde yüksek seviyeye çıkıyor. Bu şekilde kesme tespiti için gerekli kenar tetiklemesi sağlanmış oldu.&lt;br /&gt;&lt;br /&gt;Hattı ortak kullanan diğer ISA cihazları da kesme göndermek istediklerinde hattı yalnızca düşük seviyeye çekip bırakacakları için aynı anda iki cihazın hattı kullanmasında bir sorun oluşmuyor ve hat paylaşımı yapılabiliyor. Hattı ne kadar süre düşük seviyede tutacakları için de bir standart olmadığı için, bu da cihazdan cihaza değişebiliyor. Bir cihaz düşük seviyede tuttuğu esnada bir başka cihaz da düşük seviyeye getirip hattı sürmeyi bırakabilir örneğin, halen ilk cihaz sürmeyi bırakmadıysa da ikinci cihazın kesmesi arada kaybolacaktır. Kenar tetiklemeli kesmelerde her zaman kesme işleyici zincirinin tamamının gezilmesi kritik aralıklarda kaybolabilecek kesmelerin işlenebilmesi sebebi ile yapılmalı.&lt;br /&gt;&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;Seviye Tetiklemeli Kesmeler (Level-Triggered Interrupts)&lt;/span&gt;&lt;br /&gt;&lt;br /&gt;Seviye tetiklemeli modda, kesmeler kontrolörün hattın düşük ve ya yüksek seviyede bulunmasına bakarak tetikleniyordu. PCI spekleri kesmelerin bu şekilde olması gerekliliği koşulunu getiriyor. PCI cihazlarda kesme üretebilecek dört ayrı pin (&lt;span style="font-weight: bold;"&gt;INTx&lt;/span&gt;#) bulunuyor. Ve bu pinler aktif-düşük olarak kuzey köprüsünün aktif-düşük pinlerine bağlanıyor. Kuzey köprüsüne göre kesme girdi sayısı farklılık gösteriyor. Fakat köprünün aktif-düşük girişlerini ortak kullanan PCI cihazlarının INTx pinleri, köprü tarafından belirli ve ya programlanabilir şekilde kesme kontrolörünün pinlerine route edilerek ve de aktif-yüksek sinyale çevrilerek iletiliyor.&lt;br /&gt;&lt;br /&gt;PCI cihaz kesme ileteceği zaman kesme hattını düşük seviyede tutuyor bu anda kesme tespit edilerek kesme işleyicisine ulaşılıyor. Buradan da cihazın isteği karşılandıktan sonra yine kesme işleyicisi tarafından cihaz hattı bırakıyor. Eğer hat halen bir cihaz tarafından aktif olarak tutuluyorsa, yeniden kesme kontrolörü kesme gönderiyor. Seviye tetiklemeli kesmelerin avantajı, hat aktif olduğu sürece tespit edilebiliyor olması. Kenar tetiklemeli kesmelerin kısa bir aralıkta tespit edilmesi gerekiyor, kesme kalıcı değil ve kritik aralıklarda da kaybolabiliyor.&lt;br /&gt;&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;Sorunlar&lt;/span&gt;&lt;br /&gt;&lt;span&gt;&lt;br /&gt;Edge-triggered kesme üreten cihazların bir kaç şekilde tasarlanabildiğini söyledik. İlk zamanlar da cihazın kesme hattını tek başına kullanılacağı düşünülerek çıkışlar "&lt;span style="font-weight: bold;"&gt;totem-pole&lt;/span&gt;" ya da "&lt;span style="font-weight: bold;"&gt;tri-state buffer&lt;/span&gt;" ile sürülmüş. Hat üzerinde bu şekilde tasarlanan birden fazla cihaz bulunursa ve her bir cihaz hattı aynı anda sürmeye kalkarsa veriyolu üzerinde kavgaya &lt;span style="font-weight: bold;"&gt;"bus fight"&lt;/span&gt; tutuşacaklardır. Belki kesme çıkışlarında "tri-state buffer" lar kullanan ama aynı anda kesme üretimi yapmayacağı garanti olan cihazlar hattı sürmedikleri zamanlarda çıkışlarını sürmeyerek yüksek empedansda bırakabilirler. Bu şekilde bir ihtimal paylaşımları mümkün olabilir.&lt;/span&gt;&lt;br /&gt;&lt;span&gt;&lt;br /&gt;Edge-triggered kesme üreten cihazların çıkışları "&lt;span style="font-weight: bold;"&gt;open-collector&lt;/span&gt;" yapıldığı takdirde ve hat üzerinde bir pullup direnci ile her zaman yüksek seviyede tutulması ancak kesme üretileceği zaman düşük seviyede bir süre tutularak hattı bıraktığı anda hattın yeniden yüksek seviyeye çıkması ile (&lt;span style="font-weight: bold;"&gt;"pullup direnci ile"&lt;/span&gt;) tetikleme yapılabilir. Bunun dışında çıkışta "tri-state buffer" kullanılarak da paylaşım sağlanabilir. Hat yalnızca &lt;/span&gt;&lt;span&gt;ya düşük seviyede sürülür ya da &lt;/span&gt;&lt;span&gt;yüksek empedansda tutulur. ("&lt;span style="font-weight: bold;"&gt;float&lt;/span&gt;") Bu şekilde tasarlanmış bir ISA cihaz örneğini şu adreste &lt;a href="http://www.cryogenius.com/hardware/isarng/"&gt;bulabilirsiniz&lt;/a&gt;.&lt;br /&gt;&lt;br /&gt;&lt;/span&gt;&lt;span&gt;PC XT mimarisi zamanında kullanılmaya başlanan ISA veriyolu yapısı başta kesme paylaşımına olanak verecek şekilde tasarlanmamış. Cihazların kesme süren kısımlarının da belirli bir standardı bulunmadığı için üreticiler farklı yöntemlerle cihazlar üretmişler. Bu yüzden &lt;/span&gt;&lt;span&gt;kenar tetiklemeli ISA cihazlarda kesme paylaşımı pratikte mümkün olabilse de, çoğunlukla paylaşım yapılamıyor demek daha doğru. Özellikle anakart üzerinde sistem saati gibi sabit legacy kesmelerin paylaşımı mümkün değil. Sistem saati direk kesme kontrolörüne bağlı.&lt;/span&gt;&lt;span&gt;&lt;/span&gt;&lt;br /&gt;&lt;span&gt;&lt;br /&gt;&lt;/span&gt;&lt;span&gt;Aynı üreticinin ürettiği ve yaklaşık aynı işi yapan fakat tasarım kararları sebebiyle paylaştırılamayacak iki ISA cihazın "&lt;span style="font-weight: bold;"&gt;hardware hacking&lt;/span&gt;" ile paylaştırılabilir hale getirildiği hoş bir örneği şu adreste &lt;a href="http://www.fccps.cz/download/adv/frr/pcl-839/pcl-839.html"&gt;bulabilirsiniz&lt;/a&gt;. [Not: Site örnek aradığım zamanda açıktı, fakat kapanmış. Google ön belleğinden bakabilirsiniz. Fakat resimler&lt;br /&gt;yok tabi ki :( ]&lt;br /&gt;&lt;br /&gt;Seviye-tetiklemeli kesmelerde ise paylaşım mümkündü. Burada da sürücü tarafında dikkat edilmesi gereken kısımlar var. Eğer sürücüde bir hata bulunuyorsa seviye tetiklemeli kesmeler makinenin kilitlenmesine sebebiyet verebilir. Birden fazla cihazın seviye-tetiklemeli olarak hattı paylaştığını varsayalım. Her cihazın kendi üstünde bir şekilde kesmenin beklediğini işaret eden bir de yazmaç bulunduğunu düşünelim. Herhangi bir cihaz kesme ürettiğinde, kesme işleyicisi çalıştı ve paylaşılan hat için kaydedilen her bir cihazın kesme işleyicisini ("zincir") tek tek çalıştırmaya başladı. Zincirdeki her cihazın kesme işleyicisi, cihazda &lt;/span&gt;&lt;span&gt;hakikaten  &lt;/span&gt;&lt;span&gt;bekleyen bir kesme var mı diye bakacak, var ise yapması gerekenleri yapacak, kesme üreten cihaz bulunduğu ve isteği cevaplandığı için de kesme işleyici zincirine devam edilmeyecek, çünkü cihaz bulundu. Geri dönüldüğünde eğer başka bir cihaz da bu zaman içerisinde kesme üretmiş ve hattı aktif tutmaya devam ediyorsa, kesmeler de seviye tetiklemeli olduğundan hemen tespit edilecek ve yeniden kesme işleyicisi zincirine dönülerek, tek tek cihaz bulunmaya çalışılacak. Ki bulunarak isteği cevaplansın ve kesme üretmesi durdurulsun. İsteği cevaplanana kadar hattı aktif tutacak.&lt;br /&gt;&lt;br /&gt;Düşünün ki zincirdeki bir kesme işleyicisi bir hata sonucunda tamam bu kesme bana ait dedi ve ben bunu hallettim dedi. Halbuki kesmeyi üreten başka bir cihaz ve bu cihazın kesme işleyicisine de girilemedi. Geri dönüldüğü anda hat aktif olmaya devam edeceğinden sürekli işlemciye kesme gelecek. İşlemci de kesmelerden başka bir iş yapamaz hale gelecek. Makine bir döngüye girerek kilitlenecek.&lt;br /&gt;&lt;/span&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/5727727405189705833-4130300564490306818?l=cekirdeknotlari.blogspot.com' alt='' /&gt;&lt;/div&gt;</content><link rel='replies' type='application/atom+xml' href='http://cekirdeknotlari.blogspot.com/feeds/4130300564490306818/comments/default' title='Post Comments'/><link rel='replies' type='text/html' href='http://www.blogger.com/comment.g?blogID=5727727405189705833&amp;postID=4130300564490306818' title='0 Comments'/><link rel='edit' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/4130300564490306818'/><link rel='self' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/4130300564490306818'/><link rel='alternate' type='text/html' href='http://cekirdeknotlari.blogspot.com/2007/02/kenar-edge-ve-seviye-level-tetiklemeli.html' title='Kenar (edge) ve seviye (level) tetiklemeli kesmeler üzerine'/><author><name>Faik Uygur</name><uri>http://www.blogger.com/profile/15466697634443584707</uri><email>noreply@blogger.com</email><gd:image rel='http://schemas.google.com/g/2005#thumbnail' width='16' height='16' src='http://img2.blogblog.com/img/b16-rounded.gif'/></author><media:thumbnail xmlns:media='http://search.yahoo.com/mrss/' url='http://3.bp.blogspot.com/_VELeD1M_mEU/ReXoT_AsZHI/AAAAAAAAAAM/Wxshpu_9ms8/s72-c/edge-level.png' height='72' width='72'/><thr:total>0</thr:total></entry><entry><id>tag:blogger.com,1999:blog-5727727405189705833.post-1112369268254470287</id><published>2007-02-17T23:45:00.000+02:00</published><updated>2007-02-19T00:36:48.463+02:00</updated><category scheme='http://www.blogger.com/atom/ns#' term='bios'/><category scheme='http://www.blogger.com/atom/ns#' term='usb'/><title type='text'>SMM, USB Legacy Desteği ve USB Handoff üzerine</title><content type='html'>&lt;div style="text-align: justify;"&gt;x86 işlemcilerin protected, real ve virtual-8086 modlarının dışında çalıştığı özel bir mod daha bulunuyor: System Management Mode (&lt;span style="font-weight: bold;"&gt;SMM&lt;/span&gt;). Bu mod, güç yönetimi, sıcaklık ve fan takibi gibi sistem kontrolüne yönelik özel işler için ve yalnızca BIOS'un kullanımına yönelik olarak tasarlanmış bir mod. ACPI öncesi bahsi geçen işler, işlemcinin bu özelliği kullanılarak yapılıyor. Mekanizma şu şekilde işliyor: İşlemci'de aynı INTR# pin'i gibi bir de &lt;span style="font-weight: bold;"&gt;SMI#&lt;/span&gt; pin'i bulunuyor. Bu pin aktif olduğu anda işlemci ne yapıyorsa yapsın o işi bırakıyor, bekleyen tüm komutların sonuçlanmasını bekliyor, write-buffer'ını boşaltıyor, işlemci durumunu sistem hafızasından ayrı bir adres uzayı olan &lt;span style="font-weight: bold;"&gt;SMRAM&lt;/span&gt;'e yazıyor ve SM mod'a geçiş yapıyor. SMRAM içerisinde gerçekleşen durum ile başa çıkacak BIOS &lt;span style="font-weight: bold;"&gt;SMI handler&lt;/span&gt; kodu çalışıyor. İş bitince yalnızca bu modda çalışan &lt;span style="font-weight: bold;"&gt;RSM&lt;/span&gt; opkodu çalıştırılıp, moddan çıkılıyor. İşletim sisteminin ruhu duymadan, sanki hiç bir şey olmamış gibi herşey kaldığı yerden devam ediyor.&lt;br /&gt;&lt;/div&gt;&lt;br /&gt;&lt;div style="text-align: justify;"&gt;SMRAM, SM modunda kullanılan özel bir hafıza bölgesi. SMRAM sistem tasarımına bağlı olarak ne uygulamaların, ne de işletim sisteminin görebileceği bir yerde bulunuyor. BIOS, POST sonrası bu modda çalışacak SMI handler'ın kod, data ve stack bölümlerini daha önceden buraya kopyalamış oluyor. Geçiş sırasında da işlemci durumu daha sonra geri dönülebilmesi için SMRAM'in belirli bir bölgesinde saklanıyor. SMRAM sistem tasarımı açısından ayrı bir hafıza modülünde olabileceği gibi sistem hafızasının bir bölümü de SMRAM olarak kullanılabiliyor. Genelde de ikinci yöntem tercih ediliyor. Bu bölgeye işletim sistemi'nin dahi erişiminin engellenmesi işi chipset tasarımı ile sağlanıyor. Genelde uygulanan yöntem 640 KB ile 1 MB arasındaki 0xA0000 adresi ile başlayıp 0xBFFFF adresine haritalanmış olan Video RAM'i ile SMRAM'in sistem hafızasında aynı bölgeye haritalanması ile sağlanıyor. Herhangi bir modda 0xA0000 - 0xBFFFF fiziksel adresi arasına erişim istendiğinde chipset Video RAM'e ulaştırırken, SMI# sonrası SMM moda geçiş yapıldığında hafıza kontrolörü SMRAM'e erişim sağlıyor.&lt;br /&gt;&lt;br /&gt;SMI kullanımına örnek olarak sistemde yerleştirilmiş olan bir sıcaklık sensörünün bağlı olduğu mikrodenetleyicinin belli bir sıcaklık seviyesine ulaştığında SMI (System Management Interrupt) ile işlemciyi uyarması ve ardından BIOS SMI handlerı çalışarak firmware seviyesinde gereken işi yapması olarak verilebilir. Kritik seviyede BIOS makineyi kapatabilir. Burada önemli nokta bu durumdan işletim sisteminin haberi olmaması. ACPI'de ise durum farklı, bu durumdan işletim sisteminin de haberi oluyor, ilgili mesajları gösterebiliyor ve ACPI işletim sistemi tarafında makine kapatılıyor.&lt;br /&gt;&lt;br /&gt;SMI'ı kullanan daha ileri seviye bir yapı daha var. O da BIOS'lardaki &lt;span style="font-weight: bold;"&gt;USB Legacy Desteği&lt;/span&gt; özelliği.Bu ileri seviye özellik, USB mouse ve klavye kullanan makinelerin USB desteği olmayan işletim sistemlerinde PS/2 mouse/klavye emülasyonu yapılması için tasarlanmış. İşletim sistemi mouse ve klavye verilerini PS/2 veri yolu üzerinden aldığını zannediyor.&lt;br /&gt;&lt;br /&gt;Tabi ki bu işin bir &lt;a href="http://www.google.com/patents?vid=USPAT6067589&amp;id=rE8EAAAAEBAJ&amp;amp;dq=usb+legacy"&gt;patenti&lt;/a&gt; var. Muhteşem &lt;a href="http://www.google.com/patents"&gt;google patents&lt;/a&gt; hizmeti sayesinde ayrıntılı olarak işleyişi görebiliriz. İşleyiş Open Host Controller Interface Speklerinin Appendix B kısmında "Legacy Support Interface Specification" başlığı altında anlatılıyor. Tabi ki patent bunun donanımsal olarak implimentasyonu üzerine.&lt;br /&gt;&lt;br /&gt;İşleyiş OHCI host controller için şu şekilde:&lt;br /&gt;&lt;br /&gt;Öncelikle standart olarak klavye ve mouse veri alışverişinin "keyboard controller (KBC)" üzerinden 0x60 ve 0x64 portları aracılığı ile gerçekleştiğini bilmemiz gerekiyor. Yine PC mimarisinde standart olarak klavye için IRQ1 ve mouse için IRQ12 interruptlarının kullanıldığını bilmeliyiz. Sisteminizde /proc/interrupts dosyasına baktığınızda bu iki interrupt'ın da i8042 tarafından kullanıldığını göreceksiniz. Şu &lt;a href="http://my.execpc.com/%7Egeezer/osd/kbd/kbd.txt"&gt;adreste&lt;/a&gt; i8042 yazmaçları için güzel bir özet bulunuyor. Burada daha fazla detaylandırmak anlamsız. Çekirdekte düşük seviye i8042 sürücüsü ise drivers/input/serio/i8042.c altında bulunuyor.&lt;br /&gt;&lt;br /&gt;Emülasyon modunda 0x60 ve 0x64 portlarına okuma ve yazmalar host controller üzerindeki yazmaçlara gerçekleşiyor. USB legacy desteği açık ise ve USB klavye ve ya mouse takılı ise host controller usb paketlerini yakaladığı anda işlemciye SMI gönderiyor, SM moduna giren işlemci'de &lt;span style="font-weight: bold;"&gt;BIOS SMI handler&lt;/span&gt;'da USB paketlerini yorumlayarak PS/2 datasına çeviriyor ve bu portlardan erişilecek şekilde host controller üzerinde ilgili yere yazıyor. Portlara herhangi bir yazma işlemi yapıldığında host controller yakalandığı için yine bir SMI göndererek BIOS'a girilip PS/2 data USB paketlerine çevrilip host controller'a iletiliyor. Host controllerda USB paketlerini cihazlara gönderiyor.&lt;br /&gt;&lt;br /&gt;Buradaki ilginç nokta eğer USB emulasyonu açık ise ve sisteminizde hiç USB cihaz bağlı değil, yalnızca PS/2 klavye ve mouse kullanıyor olsanız bile herşey yine host controller emulasyonu ile gerçekleşiyor. Bunun için asıl KBC arayüzünün oluşturduğu interruptlar IRQ1 ve IRQ12 host controller üzerinden geçiriliyor. Yani asıl interrupt controller ile interrupt üreticisi arasında bulunuyor. Yakaladığı anda biliyor ki KBC de PS/2 datası bulunuyor. Bunları alıp emülasyonda asıl kendi üzerinden erişilen portların bulunduğu bölgeye ilgili PS/2 verilerini yerleştiriyor ve ardından ilgili interrupt'ı gönderiyor. İşletim sistemi interruptı yakalandıktan sonra verileri host controller'ın legacy portları (0x60 ve 0x64) üzerinden okunuyor. Biraz saçma gibi gelebilir. Ortada USB filan yokken neden böyle bir şeye gerek var. Sebebi USB'nin de olabileceği. Yani USB mouse ile PS/2 klavye'yi birlikte kullanabilmelisiniz. Bu yüzden emülasyon modundan herşey host controller üzerinden geçiyor. Bu emulasyon yapısı'nın OHCI için geçerli olduğunu belirtelim, UHCI ile benzer ancak ufak farklar içeren bir şekilde emulasyon işi gerçekleşiyor. Tüm host controller'ların speklerinde ilgili bölümlerde ayrıntılı bilgi bulunuyor.&lt;br /&gt;&lt;br /&gt;USB mouse ve klavye paketleri host controller'a geldikten sonra işlemciye SMI gönderiyor, böylece emulasyon yapılıyor dedik. Emulasyonun amacı USB desteği olmayan ancak PS/2 desteği olan işletim sistemleri içindi. O zaman USB desteği olan işletim sistemlerinde &lt;span style="font-weight: bold;"&gt;host controller sürücüsü&lt;/span&gt;&lt;span style="font-weight: bold;"&gt;'nün&lt;/span&gt; bir noktada USB interruptlarını karşılıyor olması gerekiyor. İşte bu işleme BIOS'un işletim sistemine USB devri (&lt;span style="font-weight: bold;"&gt;"USB Handoff"&lt;/span&gt;) ismi veriliyor. OpenHCI için konuşacak olursak, host controller basit bir şekilde &lt;span style="font-weight: bold;"&gt;HcControl&lt;/span&gt; yazmacında &lt;span style="font-weight: bold;"&gt;InterruptRouting&lt;/span&gt; bitini 1 ve ya 0 yaparak sırasıyla interruptları SMI aracılığı ile işlemciye ya da sistem interrupt kontrolörüne iletmeye başlıyor. Sistem interrupt kontrolörü işlemeye başladıktan sonra artık BIOS değil işletim sistemi USB paketlerini işliyor. InterruptRouting biti yalnızca okunabilir bir bit. Bu biti değiştirmek için önce &lt;span style="font-weight: bold;"&gt;HcInterruptStatus&lt;/span&gt; yazmacında OC (&lt;span style="font-weight: bold;"&gt;OwnershipChange&lt;/span&gt;) bitini set etmek daha sonra da &lt;span style="font-weight: bold;"&gt;HcControlStatus&lt;/span&gt; yazmacının OCR (&lt;span style="font-weight: bold;"&gt;OwnershipChangeRequest&lt;/span&gt;) bitini set etmek gerekiyor. Eğer InterruptRouting biti 0 olmuş ise artık interruptlar sistem kontrolörüne gidecek ve işletim sistemi yönetecek demektir. Handoff bir sebeple gerçekleşmez ise USB hiç bir cihaz çalışmayacak. InterruptRouting bitinin 0 olmadığı sürece host controller tüm interruptlerı SMI'a route edecek. Çekirdeğe baktığımızda bu işin şu şekilde yapıldığını görüyoruz:&lt;br /&gt;&lt;/div&gt;&lt;br /&gt;&lt;pre&gt;&lt;br /&gt;[drivers/usb/host/ohci-hcd.c]&lt;br /&gt;&lt;br /&gt;#ifndef IR_DISABLE&lt;br /&gt;/* SMM owns the HC?  not for long! */&lt;br /&gt;if (!no_handshake &amp;&amp;amp; ohci_readl (ohci,&lt;br /&gt;                                 &amp;ohci-&gt;regs-&gt;control) &amp;amp; OHCI_CTRL_IR) {&lt;br /&gt;    u32 temp;&lt;br /&gt;&lt;br /&gt;    ohci_dbg (ohci, "USB HC TakeOver from BIOS/SMM\n");&lt;br /&gt;&lt;br /&gt;    /* this timeout is arbitrary.  we make it long, so systems&lt;br /&gt;     * depending on usb keyboards may be usable even if the&lt;br /&gt;     * BIOS/SMM code seems pretty broken.&lt;br /&gt;     */&lt;br /&gt;     temp = 500; /* arbitrary: five seconds */&lt;br /&gt;&lt;br /&gt;     ohci_writel (ohci, OHCI_INTR_OC, &amp;ohci-&gt;regs-&gt;intrenable);&lt;br /&gt;     ohci_writel (ohci, OHCI_OCR, &amp;ohci-&gt;regs-&gt;cmdstatus);&lt;br /&gt;     while (ohci_readl (ohci, &amp;ohci-&gt;regs-&gt;control) &amp;amp; OHCI_CTRL_IR) {&lt;br /&gt;             msleep (10);&lt;br /&gt;             if (--temp == 0) {&lt;br /&gt;                 ohci_err (ohci, "USB HC takeover failed!"&lt;br /&gt;                           "  (BIOS/SMM bug)\n");&lt;br /&gt;                 return -EBUSY;&lt;br /&gt;             }&lt;br /&gt;     }&lt;br /&gt;     ohci_usb_reset (ohci);&lt;br /&gt;}&lt;br /&gt;#endif&lt;br /&gt;&lt;/pre&gt;&lt;br /&gt;&lt;br /&gt;&lt;div style="text-align: justify;"&gt;SMM özel bir mod. BIOS'daki herhangi bir hata ile makine hard freeze olabilir. Documentation/i386/usb-legacy-support.txt eski bir belge ama spesifik makinelerde legacy usb ile ilgili karşılaşılan bazı sorunlar dokümante edilmiş. Daha fazla örnek olsa güzel olurmuş.&lt;br /&gt;&lt;br /&gt;&lt;a href="http://support.microsoft.com/kb/250635"&gt;Burada&lt;/a&gt; da Windows 98 işletim sistemine özgü olduğu belirtilmiş bir race'in yol açtığı sorundan bahsediliyor. PCI enumeration sırasında işletim sistemleri, cihazların "configuration space"'inde bulunan &lt;span style="font-weight: bold;"&gt;BAR&lt;/span&gt;'lara belli bir pattern yazarak, daha önce BIOS'un haritaladığı adres'in ne uzunlukta olduğu bilgisini alıyor. /proc/iomem altında görünen başlangıç adresi BAR'dan alınırken, boyut da aynı şekilde bu pattern yazılarak alınıyor ve eklenerek bittiği adres bulunuyor. Windows 98'de karşılaşılan race durumunda ise işte tam bu esnada SMI gelir ve BIOS'a girilirse normalde controller yazmaçlarına eriştiği adres olan BAR'daki adres hatalı olacak ve bu da SMI#'ın sürekli aktif olması ile sonuçlanacak denilmiş. Yanlış isem düzeltiniz: SMI kaynağı, chipsetlerde belli bitlerin aktif hale gelmesi ile anlaşılıyor. Chipset aktif bit gördüğü anda SMI üretiyor. &lt;span style="font-weight: bold;"&gt;[Güncelleme: &lt;/span&gt;Intel 82801AA (ICH) datasheetinden bir örnek verirsek: &lt;span style="font-weight: bold;"&gt;5.12.6.1&lt;/span&gt; de &lt;span style="font-weight: bold;"&gt;THRM#&lt;/span&gt; sinyali aktif olduğu anda &lt;span style="font-weight: bold;"&gt;THRM_STS&lt;/span&gt; biti 1 yapılıyor. Eğer &lt;span style="font-weight: bold;"&gt;THRM_EN&lt;/span&gt; biti de aktif ise ikisinin aktif olduğu an SMI üretiliyor.&lt;span style="font-weight: bold;"&gt;] &lt;/span&gt;SMM'de işini bitirince bu biti temizliyor. Temizlemeden handler'dan çıkılırsa sürekli SMI üretilip duruluyor çünkü bit aktif ve sanki yeni bir SMI'mış gibi algılanıyor. Bundan sonra OHCI spekinin &lt;span style="font-weight: bold;"&gt;5.1.13.3&lt;/span&gt; bölümünde anlatıldığı üzere OS OwnershipChangeRequest bitini set ettiği anda &lt;span style="font-weight: bold;"&gt;SMM&lt;/span&gt; bundan haberdar olup temizlik işlerini yapıp InterruptRouting bitini temizliyor. Böylece handoff gerçekleşiyor. Ancak adresteki raporda belirtildiğine göre SMI'ın sürekli aktif olması sebebiyle bu request SMM tarafından karşılanmıyor ve handoff gerçekleşmiyor. Bu da Windows 98 makinede USB cihazların algılanamaası ile sonuçlanıyor.&lt;/div&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/5727727405189705833-1112369268254470287?l=cekirdeknotlari.blogspot.com' alt='' /&gt;&lt;/div&gt;</content><link rel='replies' type='application/atom+xml' href='http://cekirdeknotlari.blogspot.com/feeds/1112369268254470287/comments/default' title='Post Comments'/><link rel='replies' type='text/html' href='http://www.blogger.com/comment.g?blogID=5727727405189705833&amp;postID=1112369268254470287' title='0 Comments'/><link rel='edit' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/1112369268254470287'/><link rel='self' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/1112369268254470287'/><link rel='alternate' type='text/html' href='http://cekirdeknotlari.blogspot.com/2007/02/smm-usb-legacy-destei-ve-usb-handoff.html' title='SMM, USB Legacy Desteği ve USB Handoff üzerine'/><author><name>Faik Uygur</name><uri>http://www.blogger.com/profile/15466697634443584707</uri><email>noreply@blogger.com</email><gd:image rel='http://schemas.google.com/g/2005#thumbnail' width='16' height='16' src='http://img2.blogblog.com/img/b16-rounded.gif'/></author><thr:total>0</thr:total></entry><entry><id>tag:blogger.com,1999:blog-5727727405189705833.post-46027710524536557</id><published>2007-02-15T19:50:00.000+02:00</published><updated>2007-02-15T23:53:19.218+02:00</updated><category scheme='http://www.blogger.com/atom/ns#' term='pci'/><title type='text'>PCI Latency Timer, MAX_LAT ve MIN_GNT</title><content type='html'>&lt;div style="text-align: justify;"&gt;PCI cihaz bus'a erişmek istediği anda &lt;span style="font-weight: bold;"&gt;REQ#&lt;/span&gt; hattı üzerinden &lt;span style="font-weight: bold;"&gt;arbiter&lt;/span&gt;'a istekte bulunuyor. Arbiter'da PCI speklerinde implimentasyona bırakılan adil bir şekilde cihazların isteklerini &lt;span style="font-weight: bold;"&gt;GNT#&lt;/span&gt; hattını aktif hale getirerek karşılıyor. Eğer GNT# aktif hale getirildikten 16 PCI clock sonra bir işlem başlamaz ise, arbiter, cihazı "kırık" olarak kabul edip sonraki istekleri gözardı edebilir.&lt;br /&gt;&lt;br /&gt;Cihaz GNT# aldıktan sonra adres fazı sonrası, data fazına başlayacak. Cihaz izni alana kadar diğer cihazların devam eden işlemlerinin bitmesini bekledi, arbiter'ın kendisine bus'ı vermesini bekledi. Eğer bu kadar zaman bekledikten sonra, bus'ı alıp da, DWORD boyutunda tek bir veri transfer işlemi yapacak olsa cihaz açısından performans oldukça düşük olacaktır. Bunun için veri transferi &lt;span style="font-weight: bold;"&gt;"burst"&lt;/span&gt; modda gerçekleştiriliyor. Başlayan transferin veri fazı ardından ardışıl adreslere veri transferi devam ediyor.&lt;br /&gt;&lt;br /&gt;Peki nereye kadar? Bunu belirlemek için PCI &lt;span style="font-weight: bold;"&gt;"bus-mastering"&lt;/span&gt; özelliğine sahip ve iki data fazından fazla burst edebilecek cihazların bir zamanlayıcı barındırması gerekiyor. Bunun ismi &lt;span style="font-weight: bold;"&gt;"Latency Timer"&lt;/span&gt;. Timer'ın sayacağı değer ise cihazın &lt;span style="font-weight: bold;"&gt;"configuration space"&lt;/span&gt;'inde saklı. Her transfer başlangıcında (&lt;span style="font-weight: bold;"&gt;FRAME#&lt;/span&gt; aktif hale getirildiğinde) timer yeniden başlıyor. Timer sona erdiğinde cihaz işlemi sonlandırıyor. (Aslında zamanlayıcı sonlandırma için tek parametre değil. O an yürütülen PCI işlem cinsine göre de gelebilen bir takım parametreler daha var. Ayrıntılar spekte ve yazılımsal açıdan baktığımızda çok da önemli değil.) Daha önce de işini bitirebilir ama timer dolduğunda bus'ı bırakmak zorunda.&lt;br /&gt;&lt;br /&gt;Genel olarak baktığımızda burada ortaya çıkan "Latency Timer"'ın cihazın ne kadar bus'a sahip olacağı üzerinde önemli bir etkisinin olması. Eğer bu değer çok kısa olursa, diğer bekleyen cihazların gecikmesi azalıyor, eğer fazla olursa daha fazla veri gönderimi yapılıp, cihaz açısından performans artışı oluyor. &lt;span style="font-weight: bold;"&gt;"Latency"&lt;/span&gt; ve &lt;span style="font-weight: bold;"&gt;"Throughput"&lt;/span&gt; dengesi söz konusu.&lt;br /&gt;&lt;br /&gt;Peki bu değeri kim ve neye göre belirliyor? Değeri asıl belirleyen BIOS. Açılış sırasında bu değeri "configuration space"'de tüm cihazların ilgili offsetlerine yazıyor. Bunun dışında nadiren çekirdek sürücüleri de tanıdığı cihaz için bu değer ile oynayarak bir nevi "fine tuning" yapabiliyor. Değeri root hakları ile kullanıcı da &lt;span style="font-weight: bold;"&gt;"setpci"&lt;/span&gt; komutu aracılığı ile yazabilir. Çok nadir durumlarda &lt;a href="http://www-128.ibm.com/developerworks/library/l-hw2.html"&gt;ses kartı ile ilgili&lt;/a&gt; ve ya &lt;a href="http://www.mythtv.org/wiki/index.php/PCI_Latency"&gt;görüntü yakalama kartları ile ilgili&lt;/a&gt; gecikme problemlerini çözmek için kullanılabilir. Ama değerle oynamak çoğu zaman gereksiz. "Latency timer" register'ı "configuration space" 0Dh offsetinde bulunuyor.&lt;br /&gt;&lt;br /&gt;faik@iago ~ $ /usr/sbin/lspci -s 0a:08.0 -xxx&lt;br /&gt;0a:08.0 Ethernet controller: Intel Corporation PRO/100 VE Network Connection (rev 02)&lt;br /&gt;00: 86 80 92 10 07 00 90 02 02 00 00 02 10 &lt;span style="font-weight: bold;"&gt;40&lt;/span&gt; 00 00&lt;br /&gt;10: 00 50 00 d2 01 60 00 00 00 00 00 00 00 00 00 00&lt;br /&gt;20: 00 00 00 00 00 00 00 00 00 00 00 00 4d 10 ef 81&lt;br /&gt;30: 00 00 00 00 dc 00 00 00 00 00 00 00 0b 01 &lt;span style="font-weight: bold;"&gt;08 38&lt;/span&gt;&lt;br /&gt;&lt;br /&gt;faik@iago ~ $ sudo /usr/sbin/setpci -s 0a:08.0 latency_timer=80&lt;br /&gt;&lt;br /&gt;faik@iago ~ $ /usr/sbin/lspci -s 0a:08.0 -xxx&lt;br /&gt;0a:08.0 Ethernet controller: Intel Corporation PRO/100 VE Network Connection (rev 02)&lt;br /&gt;00: 86 80 92 10 07 00 90 02 02 00 00 02 10 &lt;span style="font-weight: bold;"&gt;80&lt;/span&gt; 00 00&lt;br /&gt;10: 00 50 00 d2 01 60 00 00 00 00 00 00 00 00 00 00&lt;br /&gt;20: 00 00 00 00 00 00 00 00 00 00 00 00 4d 10 ef 81&lt;br /&gt;30: 00 00 00 00 dc 00 00 00 00 00 00 00 0b 01 &lt;span style="font-weight: bold;"&gt;08 38&lt;/span&gt;&lt;br /&gt;&lt;br /&gt;Bu değer cihazın ihtiyaçlarına göre belirlenmesi gereken bir değer. Cihazın ihtiyaçlarını da BIOS'un ya da çekirdeğin anlayabilmesi için cihaz'ın "configuration space"'inde tanımlı iki adet register mevcut. Bunlar &lt;span style="font-weight: bold;"&gt;MAX_LAT&lt;/span&gt; ve &lt;span style="font-weight: bold;"&gt;MIN_GNT&lt;/span&gt; registerları. Sırasıyla 3Fh ve 3Eh offsetlerinde bulunuyorlar.&lt;br /&gt;&lt;br /&gt;MIN_GNT bir cihazın ne uzunlukta "burst" zamanına ihtiyacı olduğunu 33 Mhz'lik PCI clock referansı ile 8 PCI clock birim cinsinden saklıyor. Sebebi aşağıdaki örnek ile daha rahat anlaşılıyor. MAX_LAT ise cihazın ne sıklıkta bus'a ihtiyacı olduğunu belirtiyor.&lt;br /&gt;&lt;br /&gt;PCI speklerinde çok güzel bir örnek verilmiş. 100 Mbs'lik bir ethernet cihazı yaptığınızı düşünün. Yaklaşık 10 MB/s lik bir "throughput" karşılamak istiyorsunuz. Ethernet'inizin de her iki yön için 64-byte boyutunda bufferları bulunuyor. Her bir buffer'ı iki 32-bytelık bölme olarak (&lt;a href="http://www.google.com/patents?vid=USPAT5224213&amp;id=prkXAAAAEBAJ&amp;amp;dq=ping-pong+data+buffer&amp;ie=ISO-8859-1"&gt;&lt;span style="font-weight: bold;"&gt;ping-pong buffer&lt;/span&gt;&lt;/a&gt;) yapmak optimum performansı sağlıyor denilmiş. Rx buffer'ı düşünün. 32-byte yazılan diğer 32-byte'lık bölmeye kaydı. O işlenirken aynı zamanda önceki 32-byte'da doldurulabiliyor. Her 32-byte transfer 8 DWORD'e tekabul ediyor. Bu da 8 veri fazı/8 clock demek. 33 Mhz için 1/4 mikro saniyelik (1/33 Mhz * 8) bir zamana denk geliyor. Birim olarak en az 8 clock alınmış. Yani MIN_GNT 1 olmalı. Eğer böyle olursa cihaz istemiş olduğu 32-bytelık "burst" transferi bu zaman içinde bir seferde gerçekleştirebilir. 10 MB/s lik "throughput" için ise 32-byte'lık buffer için 32/10 MB/s ile 3.2 mikro saniyede bir "burst" yapmak yeterli geliyor. Bu da yaklaşık MAX_LAT için 12 değerine denk geliyor. (3.2 / 0.25) MAX_LAT ve MIN_GNT yazmaçları sadece okunabiliyor. Cihaz karakterini belirliyor.&lt;br /&gt;&lt;br /&gt;Çekirdek açılışında görünen  şu mesajlar, çekirdeğin ayarladığı değeri bildiriyor:&lt;br /&gt;&lt;br /&gt;&lt;span style="font-weight: bold;"&gt;PCI: Setting latency timer of device 0000:00:1d.1 to 64&lt;br /&gt;&lt;br /&gt;&lt;/span&gt;[arch/i386/pci/i386.c]&lt;span style="font-weight: bold;"&gt;&lt;span style="font-weight: bold;"&gt;&lt;/span&gt;&lt;br /&gt;&lt;/span&gt;&lt;br /&gt;&lt;pre&gt;&lt;br /&gt;/*&lt;br /&gt;*  If we set up a device for bus mastering, we need to check the latency&lt;br /&gt;*  timer as certain crappy BIOSes forget to set it properly.&lt;br /&gt;*/&lt;span style="font-weight: bold;"&gt;&lt;br /&gt;&lt;/span&gt;unsigned int pcibios_max_latency = 255;&lt;br /&gt;&lt;br /&gt;void pcibios_set_master(struct pci_dev *dev)&lt;br /&gt;{&lt;br /&gt;    u8 lat;&lt;br /&gt;    pci_read_config_byte(dev, PCI_LATENCY_TIMER, &amp;lat);&lt;br /&gt;    if (lat &lt; 16)&lt;br /&gt;        lat = (64 &lt;= pcibios_max_latency) ? 64 : pcibios_max_latency;&lt;br /&gt;    else if (lat &gt; pcibios_max_latency)&lt;br /&gt;        lat = pcibios_max_latency;&lt;br /&gt;    else&lt;br /&gt;        return;&lt;br /&gt;    printk(KERN_DEBUG "PCI: Setting latency timer of device %s to %d\n",&lt;br /&gt;        pci_name(dev), lat);&lt;br /&gt;    pci_write_config_byte(dev, PCI_LATENCY_TIMER, lat);&lt;br /&gt;}&lt;br /&gt;&lt;/pre&gt;&lt;br /&gt;Çekirdeğin ayarlamasında anladığım iki sebep var. Birincisi comment'de belirtildiği gibi bazı BIOS'ların yanlış set etmesi ve ya hiç etmemesi. İkinci sebebi ise bazı chipsetlerdeki sorunlara workaround yapabilmek için. Örneğin bazı SiS chipsetlerinde latency timer 32'den büyük olursa bus'da kilitlenme olabiliyormuş.&lt;br /&gt;&lt;br /&gt;[arch/i386/pci/fixup.c]&lt;br /&gt;&lt;pre&gt;&lt;br /&gt;static void __devinit  pci_fixup_latency(struct pci_dev *d)&lt;br /&gt;{&lt;br /&gt;/*&lt;br /&gt;*  SiS 5597 and 5598 chipsets require latency timer set to&lt;br /&gt;*  at most 32 to avoid lockups.&lt;br /&gt;*/&lt;br /&gt;    DBG("PCI: Setting max latency to 32\n");&lt;br /&gt;    pcibios_max_latency = 32;&lt;br /&gt;}&lt;br /&gt;DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_SI, PCI_DEVICE_ID_SI_5597, pci_fixup_latency);&lt;br /&gt;DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_SI, PCI_DEVICE_ID_SI_5598, pci_fixup_latency);&lt;br /&gt;&lt;/pre&gt;&lt;br /&gt;BIOS tarafında özel bir şey var mı diye LinuxBIOS'a baktığımda öntanımlı olarak tüm timerları 0x40 olarak ayarladığını görüyorum. Hani tüm cihazlara bakılıp bir algoritmayla optimum paylaşım için değerler hesaplayıp ayrı ayrı yazma gibi bir şey yok.&lt;br /&gt;&lt;br /&gt;[src/devices/pci_device.c]&lt;br /&gt;&lt;pre&gt;&lt;br /&gt;void pci_dev_set_resources(struct device *dev)&lt;br /&gt;...&lt;br /&gt;/* set a default latency timer */&lt;br /&gt;pci_write_config8(dev, PCI_LATENCY_TIMER, 0x40);&lt;br /&gt;&lt;/pre&gt;&lt;br /&gt;Daha inceleme fırsatı bulamadığım PCI Express mimarisinde MIN_GNT ve MAX_LAT bir şey ifade etmiyor. PCI Express yapısı PCI'ya göre oldukça farklı. İnceleyince daha ayrıntılı görürüz.&lt;br /&gt;&lt;/div&gt;&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/5727727405189705833-46027710524536557?l=cekirdeknotlari.blogspot.com' alt='' /&gt;&lt;/div&gt;</content><link rel='replies' type='application/atom+xml' href='http://cekirdeknotlari.blogspot.com/feeds/46027710524536557/comments/default' title='Post Comments'/><link rel='replies' type='text/html' href='http://www.blogger.com/comment.g?blogID=5727727405189705833&amp;postID=46027710524536557' title='0 Comments'/><link rel='edit' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/46027710524536557'/><link rel='self' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/46027710524536557'/><link rel='alternate' type='text/html' href='http://cekirdeknotlari.blogspot.com/2007/02/pci-latency-timer-maxlat-ve-mingnt.html' title='PCI Latency Timer, MAX_LAT ve MIN_GNT'/><author><name>Faik Uygur</name><uri>http://www.blogger.com/profile/15466697634443584707</uri><email>noreply@blogger.com</email><gd:image rel='http://schemas.google.com/g/2005#thumbnail' width='16' height='16' src='http://img2.blogblog.com/img/b16-rounded.gif'/></author><thr:total>0</thr:total></entry><entry><id>tag:blogger.com,1999:blog-5727727405189705833.post-5007766997363152657</id><published>2007-02-14T15:01:00.000+02:00</published><updated>2007-02-15T23:22:43.050+02:00</updated><category scheme='http://www.blogger.com/atom/ns#' term='pci'/><title type='text'>PCI veri güvenilirliği üzerine</title><content type='html'>x86 mimarisine sahip makinelerde işlemcinin anladığı iki adres uzayı bulunuyor. &lt;span style="font-weight: bold;"&gt;Hafıza&lt;/span&gt; ve &lt;span style="font-weight: bold;"&gt;I/O&lt;/span&gt;. x86, &lt;span style="font-weight: bold;"&gt;I/O&lt;/span&gt; adresindeki bir PCI cihaza yazma işlemi yapıldığında işlemci devam etmeden, yazma işleminin cihaza eriştiğinden emin olabiliriz. Fakat &lt;span style="font-weight: bold;"&gt;MMIO&lt;/span&gt; adresindeki bir cihaza yazma işlemi yapıldığında işlemci devam ederken, yazmanın cihaza eriştiğinden emin olamayız. PCI cihaza yapılan yazma işlemi bir yerlerde (host bridge'de ya da ikinci bir PCI bus'a giden bir ara bridge'de) bekletiliyor olabilir. Toplu yazma yapılabileceği durumlar için bir tasarruf. Bu işleme &lt;span style="font-weight: bold;"&gt;"write-posting" &lt;/span&gt;deniyor. Eğer işlemcinin bir sonraki yapacağı işlemde cihaza önceki yazma işleminin sonuçlanmış olması gerekliliği var ise, cihaz üzerinde herhangi "etkisiz" bir yerden okuma yapılabilir. Tüm bridge'ler okuma işlemi öncesi bekletilen yazma işlemlerini tamamlıyor. Bir nevi, buffer &lt;span style="font-weight: bold;"&gt;"flush"&lt;/span&gt; ediliyor.&lt;br /&gt;&lt;br /&gt;Bunlar sürücü açısından baktığımız zaman bildiğimiz şeyler. Ama ilginç olan bir şey var ki, cihaz açısından baktığımızda garip bir durum ortaya çıkıyor. Herhangi bir cihaz, hafızada bir yere yazma işlemini bitirdikten sonra işlemciye bir interrupt göndererek bunun haberini verdiğini düşünelim. İşte burada öyle bir durum olabilir ki interrupt işlemciye, veriler hafızaya ulaşmadan erişebilir. İlginç... Bunu engellemek için PCI speklerinde üç yöntemden bahsediliyor:&lt;br /&gt;&lt;br /&gt;1. &lt;span style="font-weight: bold;"&gt;Sistem donanımı&lt;/span&gt; "write-posting" yapılan bufferları, cihaz'ın interruptı işlemciye ulaşmadan önce flush edebilir. Yani donanımsal olarak yazmalar izlenerek interrupt görüldüğünde yazmaların ulaştığını donanım garantileyecek.&lt;br /&gt;2. &lt;span style="font-weight: bold;"&gt;Cihaz&lt;/span&gt; yazma işleminin hemen ardından yazdığı veriyi okur ve ardından interrupt gönderir.&lt;br /&gt;3. &lt;span style="font-weight: bold;"&gt;Sürücü&lt;/span&gt; interrupt geldikten sonra cihazın üzerindeki bir yazmacı okur ve ardından cihazın yazdığı veriye erişir.&lt;div class="blogger-post-footer"&gt;&lt;img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/5727727405189705833-5007766997363152657?l=cekirdeknotlari.blogspot.com' alt='' /&gt;&lt;/div&gt;</content><link rel='replies' type='application/atom+xml' href='http://cekirdeknotlari.blogspot.com/feeds/5007766997363152657/comments/default' title='Post Comments'/><link rel='replies' type='text/html' href='http://www.blogger.com/comment.g?blogID=5727727405189705833&amp;postID=5007766997363152657' title='0 Comments'/><link rel='edit' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/5007766997363152657'/><link rel='self' type='application/atom+xml' href='http://www.blogger.com/feeds/5727727405189705833/posts/default/5007766997363152657'/><link rel='alternate' type='text/html' href='http://cekirdeknotlari.blogspot.com/2007/02/pci-veri-gvenilirlii-zerine.html' title='PCI veri güvenilirliği üzerine'/><author><name>Faik Uygur</name><uri>http://www.blogger.com/profile/15466697634443584707</uri><email>noreply@blogger.com</email><gd:image rel='http://schemas.google.com/g/2005#thumbnail' width='16' height='16' src='http://img2.blogblog.com/img/b16-rounded.gif'/></author><thr:total>0</thr:total></entry></feed>
