mirror of
https://github.com/Zeal-Operating-System/ZealOS.git
synced 2025-03-31 13:25:09 +01:00
Begin implementing E1000 RX/TX.
This commit is contained in:
parent
1003d78140
commit
afdc25f9b8
8 changed files with 658 additions and 278 deletions
Binary file not shown.
|
@ -136,264 +136,448 @@ body {background-color:#000000;}
|
|||
<a name="l109"></a>
|
||||
<a name="l110"></a> </span><span class=cF1>U8</span><span class=cF0> mac_address[</span><span class=cFE>6</span><span class=cF0>];
|
||||
<a name="l111"></a> </span><span class=cF9>U64</span><span class=cF0> mmio_address;
|
||||
<a name="l112"></a> </span><span class=cF1>U8</span><span class=cF0> *rx_de_buffer; </span><span class=cF2>// Uncached-alias of pointer to the buffer of RX Descriptor Entries.</span><span class=cF0>
|
||||
<a name="l113"></a> </span><span class=cF1>U8</span><span class=cF0> *tx_de_buffer; </span><span class=cF2>// Uncached-alias of pointer to the buffer of TX Descriptor Entries.</span><span class=cF0>
|
||||
<a name="l114"></a> </span><span class=cF1>U8</span><span class=cF0> *rx_de_buffer_phys; </span><span class=cF2>// Pointer to the buffer of RX Descriptor Entries. (Code Heap, lower 2Gb)</span><span class=cF0>
|
||||
<a name="l115"></a> </span><span class=cF1>U8</span><span class=cF0> *tx_de_buffer_phys; </span><span class=cF2>// Pointer to the buffer of TX Descriptor Entries. (Code Heap, lower 2Gb)</span><span class=cF0>
|
||||
<a name="l116"></a> </span><span class=cF9>U64</span><span class=cF0> rx_buffer_addr; </span><span class=cF2>// Uncached-alias of address of receive buffers.</span><span class=cF0>
|
||||
<a name="l117"></a> </span><span class=cF9>U64</span><span class=cF0> tx_buffer_addr; </span><span class=cF2>// Uncached-alias of address of transmit buffers.</span><span class=cF0>
|
||||
<a name="l118"></a> </span><span class=cF9>U64</span><span class=cF0> rx_buffer_addr_phys; </span><span class=cF2>// Physical address of actual receive buffers (< 4 Gb)</span><span class=cF0>
|
||||
<a name="l119"></a> </span><span class=cF9>U64</span><span class=cF0> tx_buffer_addr_phys; </span><span class=cF2>// Physical address of actual transmit buffers (< 4 Gb)</span><span class=cF0>
|
||||
<a name="l112"></a>
|
||||
<a name="l113"></a> </span><span class=cF9>I64</span><span class=cF0> current_rx_de_index; </span><span class=cF2>// Current Receive DE being processed. Gets incremented, wrapped to 0 at max of E1000_RX_BUFF_COUNT.</span><span class=cF0>
|
||||
<a name="l114"></a> </span><span class=cF9>I64</span><span class=cF0> current_tx_de_index; </span><span class=cF2>// Current Transmit DE being processed. Gets incremented, wrapped to 0 at max of E1000_TX_BUFF_COUNT.</span><span class=cF0>
|
||||
<a name="l115"></a>
|
||||
<a name="l116"></a> </span><span class=cF1>U8</span><span class=cF0> *rx_de_buffer; </span><span class=cF2>// Uncached-alias of pointer to the buffer of RX Descriptor Entries.</span><span class=cF0>
|
||||
<a name="l117"></a> </span><span class=cF1>U8</span><span class=cF0> *tx_de_buffer; </span><span class=cF2>// Uncached-alias of pointer to the buffer of TX Descriptor Entries.</span><span class=cF0>
|
||||
<a name="l118"></a> </span><span class=cF1>U8</span><span class=cF0> *rx_de_buffer_phys; </span><span class=cF2>// Pointer to the buffer of RX Descriptor Entries. (Code Heap, lower 2Gb)</span><span class=cF0>
|
||||
<a name="l119"></a> </span><span class=cF1>U8</span><span class=cF0> *tx_de_buffer_phys; </span><span class=cF2>// Pointer to the buffer of TX Descriptor Entries. (Code Heap, lower 2Gb)</span><span class=cF0>
|
||||
<a name="l120"></a>
|
||||
<a name="l121"></a>} e1000; </span><span class=cF2>// e1000 is the global variable we store all of this into.</span><span class=cF0>
|
||||
<a name="l122"></a>
|
||||
<a name="l123"></a></span><span class=cF9>CPCIDev</span><span class=cF0> *E1000PCIDevFind()
|
||||
<a name="l124"></a>{</span><span class=cF2>// Find and return E1000 card as a CPCIDev pointer.</span><span class=cF0>
|
||||
<a name="l121"></a> </span><span class=cF9>U64</span><span class=cF0> rx_buffer_addr; </span><span class=cF2>// Uncached-alias of address of receive buffers.</span><span class=cF0>
|
||||
<a name="l122"></a> </span><span class=cF9>U64</span><span class=cF0> tx_buffer_addr; </span><span class=cF2>// Uncached-alias of address of transmit buffers.</span><span class=cF0>
|
||||
<a name="l123"></a> </span><span class=cF9>U64</span><span class=cF0> rx_buffer_addr_phys; </span><span class=cF2>// Physical address of actual receive buffers (< 4 Gb)</span><span class=cF0>
|
||||
<a name="l124"></a> </span><span class=cF9>U64</span><span class=cF0> tx_buffer_addr_phys; </span><span class=cF2>// Physical address of actual transmit buffers (< 4 Gb)</span><span class=cF0>
|
||||
<a name="l125"></a>
|
||||
<a name="l126"></a> </span><span class=cF9>CPCIDev</span><span class=cF0> *pci = </span><span class=cF5>PCIDevFind</span><span class=cF0>(</span><span class=cF3>PCIC_NETWORK</span><span class=cF0>,, PCIV_E1000);
|
||||
<a name="l127"></a>
|
||||
<a name="l128"></a> </span><span class=cF1>if</span><span class=cF0> (!pci)
|
||||
<a name="l129"></a> </span><span class=cF1>return</span><span class=cF0> </span><span class=cF3>NULL</span><span class=cF0>;
|
||||
<a name="l130"></a>
|
||||
<a name="l131"></a> </span><span class=cF5>ClassRep</span><span class=cF0>(pci);
|
||||
<a name="l132"></a>
|
||||
<a name="l133"></a> </span><span class=cF1>switch</span><span class=cF0> (pci->device_id)
|
||||
<a name="l134"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l135"></a> </span><span class=cF1>case</span><span class=cF0> PCID_82545EM:
|
||||
<a name="l136"></a> </span><span class=cF1>break</span><span class=cF0>;
|
||||
<a name="l137"></a>
|
||||
<a name="l138"></a> </span><span class=cF1>default</span><span class=cF0>:
|
||||
<a name="l139"></a> pci = </span><span class=cF3>NULL</span><span class=cF0>;
|
||||
<a name="l140"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l141"></a>
|
||||
<a name="l142"></a> </span><span class=cF1>return</span><span class=cF0> pci;
|
||||
<a name="l143"></a>}
|
||||
<a name="l144"></a>
|
||||
<a name="l145"></a></span><span class=cF9>U32</span><span class=cF0> E1000MMIORead(</span><span class=cF9>U32</span><span class=cF0> </span><span class=cF1>offset</span><span class=cF0>)
|
||||
<a name="l146"></a>{
|
||||
<a name="l147"></a> </span><span class=cF9>U32</span><span class=cF0> *val = e1000.mmio_address + </span><span class=cF1>offset</span><span class=cF0>;
|
||||
<a name="l148"></a>
|
||||
<a name="l149"></a> </span><span class=cF1>return</span><span class=cF0> *val;
|
||||
<a name="l150"></a>}
|
||||
<a name="l151"></a>
|
||||
<a name="l152"></a></span><span class=cF1>U0</span><span class=cF0> E1000MMIOWrite(</span><span class=cF9>U32</span><span class=cF0> </span><span class=cF1>offset</span><span class=cF0>, </span><span class=cF9>U32</span><span class=cF0> val)
|
||||
<a name="l153"></a>{
|
||||
<a name="l154"></a> </span><span class=cF9>U32</span><span class=cF0> *addr = e1000.mmio_address + </span><span class=cF1>offset</span><span class=cF0>;
|
||||
<a name="l155"></a>
|
||||
<a name="l156"></a> *addr = val;
|
||||
<a name="l157"></a>}
|
||||
<a name="l158"></a>
|
||||
<a name="l159"></a></span><span class=cF9>U16</span><span class=cF0> E1000EEPROMRead(</span><span class=cF1>U8</span><span class=cF0> word)
|
||||
<a name="l160"></a>{ </span><span class=cF2>// word arg is which U16 to read</span><span class=cF0>
|
||||
<a name="l161"></a> </span><span class=cF9>U16</span><span class=cF0> data;
|
||||
<a name="l162"></a> </span><span class=cF9>U32</span><span class=cF0> temp;
|
||||
<a name="l163"></a>
|
||||
<a name="l164"></a> E1000MMIOWrite(E1000_REG_EERD, </span><span class=cFE>1</span><span class=cF0> | word << </span><span class=cFE>8</span><span class=cF0>);
|
||||
<a name="l165"></a>
|
||||
<a name="l166"></a> </span><span class=cF1>while</span><span class=cF0> (!</span><span class=cF7>(</span><span class=cF0>(temp = E1000MMIORead</span><span class=cF7>(</span><span class=cF0>E1000_REG_EERD</span><span class=cF7>)</span><span class=cF0>) & (</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>4</span><span class=cF0>)</span><span class=cF7>)</span><span class=cF0>)
|
||||
<a name="l167"></a> </span><span class=cF5>Sleep</span><span class=cF0>(</span><span class=cFE>1</span><span class=cF0>);
|
||||
<a name="l168"></a>
|
||||
<a name="l169"></a> data = (temp >> </span><span class=cFE>16</span><span class=cF0>) & </span><span class=cFE>0xFFFF</span><span class=cF0>;
|
||||
<a name="l170"></a> </span><span class=cF1>return</span><span class=cF0> data;
|
||||
<a name="l171"></a>}
|
||||
<a name="l172"></a>
|
||||
<a name="l173"></a></span><span class=cF1>U0</span><span class=cF0> E1000MACGet()
|
||||
<a name="l174"></a>{
|
||||
<a name="l175"></a> </span><span class=cF9>I64</span><span class=cF0> i;
|
||||
<a name="l176"></a> </span><span class=cF9>U16</span><span class=cF0> mac;
|
||||
<a name="l177"></a>
|
||||
<a name="l178"></a> NetLog(</span><span class=cF6>"E1000 GET MAC: Getting VM MAC."</span><span class=cF0>);
|
||||
<a name="l179"></a>
|
||||
<a name="l180"></a> </span><span class=cF1>for</span><span class=cF0> (i = </span><span class=cFE>0</span><span class=cF0>; i < </span><span class=cFE>3</span><span class=cF0>; i++)
|
||||
<a name="l181"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l182"></a> mac = E1000EEPROMRead(i);
|
||||
<a name="l183"></a> e1000.mac_address[</span><span class=cFE>2</span><span class=cF0>*i] = mac & </span><span class=cFE>0xFF</span><span class=cF0>;
|
||||
<a name="l184"></a> e1000.mac_address[</span><span class=cFE>2</span><span class=cF0>*i+</span><span class=cFE>1</span><span class=cF0>] = (mac >> </span><span class=cFE>8</span><span class=cF0>) & </span><span class=cFE>0xFF</span><span class=cF0>;
|
||||
<a name="l185"></a> NetLog(</span><span class=cF6>" %02X %02X"</span><span class=cF0>, mac.u8[</span><span class=cFE>0</span><span class=cF0>], mac.u8[</span><span class=cFE>1</span><span class=cF0>]);
|
||||
<a name="l186"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l187"></a>
|
||||
<a name="l188"></a>}
|
||||
<a name="l189"></a>
|
||||
<a name="l190"></a></span><span class=cF1>U0</span><span class=cF0> EthernetFrameFinish(</span><span class=cF9>I64</span><span class=cF0> de_index)
|
||||
<a name="l191"></a>{</span><span class=cF2>//Alias for driver Finish TX function.</span><span class=cF0>
|
||||
<a name="l192"></a> </span><span class=cF2>//E1000TransmitPacketFinish(de_index);</span><span class=cF0>
|
||||
<a name="l193"></a> </span><span class=cF1>no_warn</span><span class=cF0> de_index;
|
||||
<a name="l194"></a> NetErr(</span><span class=cF6>"TODO E1000"</span><span class=cF0>);
|
||||
<a name="l195"></a>}
|
||||
<a name="l196"></a>
|
||||
<a name="l197"></a></span><span class=cF2>/*</span><span class=cF0>
|
||||
<a name="l198"></a></span><span class=cF2>U0 PCIInterruptsReroute(I64 base)</span><span class=cF0>
|
||||
<a name="l199"></a></span><span class=cF2>{ // todo: comments explaining process, maybe better var names</span><span class=cF0>
|
||||
<a name="l200"></a> </span><span class=cF2>I64</span><span class=cF0> </span><span class=cF2> i;</span><span class=cF0>
|
||||
<a name="l201"></a> </span><span class=cF2>U8</span><span class=cF0> </span><span class=cF2>*da = dev.uncached_alias + IOAPIC_REG;</span><span class=cF0>
|
||||
<a name="l202"></a> </span><span class=cF2>U32 *_d = dev.uncached_alias + IOAPIC_DATA;</span><span class=cF0>
|
||||
<a name="l203"></a>
|
||||
<a name="l204"></a> </span><span class=cF2>for (i = 0; i < 4; i++)</span><span class=cF0>
|
||||
<a name="l205"></a> </span><span class=cF2>{</span><span class=cF0>
|
||||
<a name="l206"></a> </span><span class=cF2>*da = IOREDTAB + i * 2 + 1;</span><span class=cF0>
|
||||
<a name="l207"></a> </span><span class=cF2>*_d = dev.mp_apic_ids[INT_DEST_CPU] << 24;</span><span class=cF0>
|
||||
<a name="l208"></a> </span><span class=cF2>*da = IOREDTAB + i * 2;</span><span class=cF0>
|
||||
<a name="l209"></a> </span><span class=cF2>*_d = 0x4000 + base + i;</span><span class=cF0>
|
||||
<a name="l210"></a> </span><span class=cF2>}</span><span class=cF0>
|
||||
<a name="l211"></a></span><span class=cF2>}</span><span class=cF0>
|
||||
<a name="l212"></a>
|
||||
<a name="l126"></a>
|
||||
<a name="l127"></a>} e1000; </span><span class=cF2>// e1000 is the global variable we store all of this into.</span><span class=cF0>
|
||||
<a name="l128"></a>
|
||||
<a name="l129"></a></span><span class=cF9>CPCIDev</span><span class=cF0> *E1000PCIDevFind()
|
||||
<a name="l130"></a>{</span><span class=cF2>// Find and return E1000 card as a CPCIDev pointer.</span><span class=cF0>
|
||||
<a name="l131"></a>
|
||||
<a name="l132"></a> </span><span class=cF9>CPCIDev</span><span class=cF0> *pci = </span><span class=cF5>PCIDevFind</span><span class=cF0>(</span><span class=cF3>PCIC_NETWORK</span><span class=cF0>,, PCIV_E1000);
|
||||
<a name="l133"></a>
|
||||
<a name="l134"></a> </span><span class=cF1>if</span><span class=cF0> (!pci)
|
||||
<a name="l135"></a> </span><span class=cF1>return</span><span class=cF0> </span><span class=cF3>NULL</span><span class=cF0>;
|
||||
<a name="l136"></a>
|
||||
<a name="l137"></a> </span><span class=cF5>ClassRep</span><span class=cF0>(pci);
|
||||
<a name="l138"></a>
|
||||
<a name="l139"></a> </span><span class=cF1>switch</span><span class=cF0> (pci->device_id)
|
||||
<a name="l140"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l141"></a> </span><span class=cF1>case</span><span class=cF0> PCID_82545EM:
|
||||
<a name="l142"></a> </span><span class=cF1>break</span><span class=cF0>;
|
||||
<a name="l143"></a>
|
||||
<a name="l144"></a> </span><span class=cF1>default</span><span class=cF0>:
|
||||
<a name="l145"></a> pci = </span><span class=cF3>NULL</span><span class=cF0>;
|
||||
<a name="l146"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l147"></a>
|
||||
<a name="l148"></a> </span><span class=cF1>return</span><span class=cF0> pci;
|
||||
<a name="l149"></a>}
|
||||
<a name="l150"></a>
|
||||
<a name="l151"></a></span><span class=cF9>U32</span><span class=cF0> E1000MMIORead(</span><span class=cF9>U32</span><span class=cF0> </span><span class=cF1>offset</span><span class=cF0>)
|
||||
<a name="l152"></a>{
|
||||
<a name="l153"></a> </span><span class=cF9>U32</span><span class=cF0> *val = e1000.mmio_address + </span><span class=cF1>offset</span><span class=cF0>;
|
||||
<a name="l154"></a>
|
||||
<a name="l155"></a> </span><span class=cF1>return</span><span class=cF0> *val;
|
||||
<a name="l156"></a>}
|
||||
<a name="l157"></a>
|
||||
<a name="l158"></a></span><span class=cF1>U0</span><span class=cF0> E1000MMIOWrite(</span><span class=cF9>U32</span><span class=cF0> </span><span class=cF1>offset</span><span class=cF0>, </span><span class=cF9>U32</span><span class=cF0> val)
|
||||
<a name="l159"></a>{
|
||||
<a name="l160"></a> </span><span class=cF9>U32</span><span class=cF0> *addr = e1000.mmio_address + </span><span class=cF1>offset</span><span class=cF0>;
|
||||
<a name="l161"></a>
|
||||
<a name="l162"></a> *addr = val;
|
||||
<a name="l163"></a>}
|
||||
<a name="l164"></a>
|
||||
<a name="l165"></a></span><span class=cF9>U16</span><span class=cF0> E1000EEPROMRead(</span><span class=cF1>U8</span><span class=cF0> word)
|
||||
<a name="l166"></a>{ </span><span class=cF2>// word arg is which U16 to read</span><span class=cF0>
|
||||
<a name="l167"></a> </span><span class=cF9>U16</span><span class=cF0> data;
|
||||
<a name="l168"></a> </span><span class=cF9>U32</span><span class=cF0> temp;
|
||||
<a name="l169"></a>
|
||||
<a name="l170"></a> E1000MMIOWrite(E1000_REG_EERD, </span><span class=cFE>1</span><span class=cF0> | word << </span><span class=cFE>8</span><span class=cF0>);
|
||||
<a name="l171"></a>
|
||||
<a name="l172"></a> </span><span class=cF1>while</span><span class=cF0> (!</span><span class=cF7>(</span><span class=cF0>(temp = E1000MMIORead</span><span class=cF7>(</span><span class=cF0>E1000_REG_EERD</span><span class=cF7>)</span><span class=cF0>) & (</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>4</span><span class=cF0>)</span><span class=cF7>)</span><span class=cF0>)
|
||||
<a name="l173"></a> </span><span class=cF5>Sleep</span><span class=cF0>(</span><span class=cFE>1</span><span class=cF0>);
|
||||
<a name="l174"></a>
|
||||
<a name="l175"></a> data = (temp >> </span><span class=cFE>16</span><span class=cF0>) & </span><span class=cFE>0xFFFF</span><span class=cF0>;
|
||||
<a name="l176"></a> </span><span class=cF1>return</span><span class=cF0> data;
|
||||
<a name="l177"></a>}
|
||||
<a name="l178"></a>
|
||||
<a name="l179"></a></span><span class=cF1>U0</span><span class=cF0> E1000MACGet()
|
||||
<a name="l180"></a>{
|
||||
<a name="l181"></a> </span><span class=cF9>I64</span><span class=cF0> i;
|
||||
<a name="l182"></a> </span><span class=cF9>U16</span><span class=cF0> mac;
|
||||
<a name="l183"></a>
|
||||
<a name="l184"></a> NetLog(</span><span class=cF6>"E1000 MAC GET: Getting MAC Address."</span><span class=cF0>);
|
||||
<a name="l185"></a>
|
||||
<a name="l186"></a> </span><span class=cF1>for</span><span class=cF0> (i = </span><span class=cFE>0</span><span class=cF0>; i < </span><span class=cFE>3</span><span class=cF0>; i++)
|
||||
<a name="l187"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l188"></a> mac = E1000EEPROMRead(i);
|
||||
<a name="l189"></a> e1000.mac_address[</span><span class=cFE>2</span><span class=cF0>*i] = mac & </span><span class=cFE>0xFF</span><span class=cF0>;
|
||||
<a name="l190"></a> e1000.mac_address[</span><span class=cFE>2</span><span class=cF0>*i+</span><span class=cFE>1</span><span class=cF0>] = (mac >> </span><span class=cFE>8</span><span class=cF0>) & </span><span class=cFE>0xFF</span><span class=cF0>;
|
||||
<a name="l191"></a> NetLog(</span><span class=cF6>" %02X %02X"</span><span class=cF0>, mac.u8[</span><span class=cFE>0</span><span class=cF0>], mac.u8[</span><span class=cFE>1</span><span class=cF0>]);
|
||||
<a name="l192"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l193"></a>
|
||||
<a name="l194"></a>}
|
||||
<a name="l195"></a>
|
||||
<a name="l196"></a></span><span class=cF9>I64</span><span class=cF0> E1000PacketReceive(</span><span class=cF1>U8</span><span class=cF0> **packet_buffer_out, </span><span class=cF9>U16</span><span class=cF0> *packet_length_out)
|
||||
<a name="l197"></a>{
|
||||
<a name="l198"></a> </span><span class=cF9>I64</span><span class=cF0> de_index = e1000.current_rx_de_index;
|
||||
<a name="l199"></a> CE1000DescriptorEntryRX *entry = &e1000.rx_de_buffer[de_index * </span><span class=cF1>sizeof</span><span class=cF0>(CE1000DescriptorEntryRX)];
|
||||
<a name="l200"></a> </span><span class=cF1>Bool</span><span class=cF0> drop = </span><span class=cF3>FALSE</span><span class=cF0>;
|
||||
<a name="l201"></a>
|
||||
<a name="l202"></a> </span><span class=cF1>if</span><span class=cF0> (entry->length < </span><span class=cFE>60</span><span class=cF0>)
|
||||
<a name="l203"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l204"></a> NetErr(</span><span class=cF6>"E1000 PACKET RECEIVE: Short Packet"</span><span class=cF0>);
|
||||
<a name="l205"></a> drop = </span><span class=cF3>TRUE</span><span class=cF0>;
|
||||
<a name="l206"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l207"></a>
|
||||
<a name="l208"></a> </span><span class=cF1>if</span><span class=cF0> (entry->status & </span><span class=cF7>(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>1</span><span class=cF7>)</span><span class=cF0>) </span><span class=cF2>// EOP ?? TODO #define</span><span class=cF0>
|
||||
<a name="l209"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l210"></a> NetErr(</span><span class=cF6>"E1000 PACKET RECEIVE: No EOP Set"</span><span class=cF0>);
|
||||
<a name="l211"></a> drop = </span><span class=cF3>TRUE</span><span class=cF0>;
|
||||
<a name="l212"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l213"></a>
|
||||
<a name="l214"></a></span><span class=cF2>U0 E1000InterruptsSetup()</span><span class=cF0>
|
||||
<a name="l215"></a></span><span class=cF2>{</span><span class=cF0>
|
||||
<a name="l216"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>PCIInterruptsReroute(I_E1000);</span><span class=cF0>
|
||||
<a name="l217"></a> </span><span class=cF2>NetErr("TODO E1000");</span><span class=cF0>
|
||||
<a name="l218"></a></span><span class=cF2>}</span><span class=cF0>
|
||||
<a name="l219"></a></span><span class=cF2>*/</span><span class=cF0>
|
||||
<a name="l220"></a>
|
||||
<a name="l221"></a></span><span class=cF1>U0</span><span class=cF0> E1000RXInit()
|
||||
<a name="l222"></a>{
|
||||
<a name="l223"></a> </span><span class=cF9>I64</span><span class=cF0> de_index;
|
||||
<a name="l224"></a>
|
||||
<a name="l225"></a> e1000.rx_de_buffer_phys = </span><span class=cF5>CAllocAligned</span><span class=cF0>(</span><span class=cF1>sizeof</span><span class=cF7>(</span><span class=cF0>CE1000DescriptorEntryRX</span><span class=cF7>)</span><span class=cF0> * E1000_RX_BUFF_COUNT,
|
||||
<a name="l226"></a> </span><span class=cFE>16</span><span class=cF0>,
|
||||
<a name="l227"></a> </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l228"></a>
|
||||
<a name="l229"></a> e1000.rx_de_buffer = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.rx_de_buffer_phys;
|
||||
<a name="l230"></a>
|
||||
<a name="l231"></a> e1000.rx_buffer_addr_phys = </span><span class=cF5>CAlloc</span><span class=cF0>(ETHERNET_FRAME_SIZE * E1000_RX_BUFF_COUNT, </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l214"></a> </span><span class=cF1>if</span><span class=cF0> (entry->errors)
|
||||
<a name="l215"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l216"></a> NetErr(</span><span class=cF6>"E1000 PACKET RECEIVE: RX DE Error Bits Set"</span><span class=cF0>);
|
||||
<a name="l217"></a> drop = </span><span class=cF3>TRUE</span><span class=cF0>;
|
||||
<a name="l218"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l219"></a>
|
||||
<a name="l220"></a> e1000.current_rx_de_index = (e1000.current_rx_de_index + </span><span class=cFE>1</span><span class=cF0>) & (E1000_RX_BUFF_COUNT - </span><span class=cFE>1</span><span class=cF0>);
|
||||
<a name="l221"></a>
|
||||
<a name="l222"></a> </span><span class=cF1>if</span><span class=cF0> (!drop)
|
||||
<a name="l223"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l224"></a> *packet_buffer_out = entry->address;
|
||||
<a name="l225"></a> *packet_length_out = entry->length;
|
||||
<a name="l226"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l227"></a> </span><span class=cF1>else</span><span class=cF0>
|
||||
<a name="l228"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l229"></a> NetErr(</span><span class=cF6>"E1000 PACKET RECEIVE: Dropping packet."</span><span class=cF0>);
|
||||
<a name="l230"></a> de_index = -</span><span class=cFE>1</span><span class=cF0>;
|
||||
<a name="l231"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l232"></a>
|
||||
<a name="l233"></a> e1000.rx_buffer_addr = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.rx_buffer_addr_phys;
|
||||
<a name="l234"></a>
|
||||
<a name="l235"></a> </span><span class=cF2>// iterate de's and make packet buffers for each</span><span class=cF0>
|
||||
<a name="l236"></a> CE1000DescriptorEntryRX *entry = e1000.rx_de_buffer;
|
||||
<a name="l237"></a> </span><span class=cF1>for</span><span class=cF0> (de_index = </span><span class=cFE>0</span><span class=cF0>; de_index < E1000_RX_BUFF_COUNT; de_index++)
|
||||
<a name="l238"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l239"></a> entry->address = e1000.rx_buffer_addr + de_index * ETHERNET_FRAME_SIZE; </span><span class=cF2>// is this right? might need to change ?..</span><span class=cF0>
|
||||
<a name="l240"></a> </span><span class=cF2>// 01000101 MAlloc's 8208 for each DE</span><span class=cF0>
|
||||
<a name="l241"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>PCNetDescriptorEntryInit(&entry[de_index],</span><span class=cF0>
|
||||
<a name="l242"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2> pcnet.rx_buffer_addr + de_index * ETHERNET_FRAME_SIZE,</span><span class=cF0>
|
||||
<a name="l243"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2> TRUE); // TRUE for is_rx.</span><span class=cF0>
|
||||
<a name="l244"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l245"></a>
|
||||
<a name="l246"></a> </span><span class=cF2>// setup rx de ring buffer</span><span class=cF0>
|
||||
<a name="l247"></a> E1000MMIOWrite(E1000_REG_RDBAH, e1000.rx_de_buffer >> </span><span class=cFE>32</span><span class=cF0>); </span><span class=cF2>// should we be using uncached addr here ?</span><span class=cF0>
|
||||
<a name="l248"></a> E1000MMIOWrite(E1000_REG_RDBAL, e1000.rx_de_buffer & </span><span class=cFE>0xFFFFFFFF</span><span class=cF0>);
|
||||
<a name="l249"></a>
|
||||
<a name="l250"></a> </span><span class=cF2>// set receive buffer length</span><span class=cF0>
|
||||
<a name="l251"></a> E1000MMIOWrite(E1000_REG_RDLEN, E1000_RX_BUFF_COUNT * </span><span class=cFE>16</span><span class=cF0>);
|
||||
<a name="l252"></a>
|
||||
<a name="l253"></a> </span><span class=cF2>// set head tail pointers</span><span class=cF0>
|
||||
<a name="l254"></a> E1000MMIOWrite(E1000_REG_RDH, </span><span class=cFE>0</span><span class=cF0>);
|
||||
<a name="l255"></a> E1000MMIOWrite(E1000_REG_RDT, E1000_RX_BUFF_COUNT);
|
||||
<a name="l256"></a>
|
||||
<a name="l257"></a> </span><span class=cF2>// set receive control reg</span><span class=cF0>
|
||||
<a name="l258"></a> E1000MMIOWrite(E1000_REG_RCTL, E1000_RCTLF_SBP |
|
||||
<a name="l259"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_UPE</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l260"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_RDMTS_HALF</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l261"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_BSIZE_2048</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l262"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_MPE</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l263"></a> E1000_RCTLF_SECRC |
|
||||
<a name="l264"></a> E1000_RCTLF_LPE |
|
||||
<a name="l265"></a> E1000_RCTLF_BAM);
|
||||
<a name="l266"></a>}
|
||||
<a name="l267"></a>
|
||||
<a name="l268"></a></span><span class=cF1>U0</span><span class=cF0> E1000TXInit()
|
||||
<a name="l269"></a>{
|
||||
<a name="l270"></a> e1000.tx_de_buffer_phys = </span><span class=cF5>CAllocAligned</span><span class=cF0>(</span><span class=cF1>sizeof</span><span class=cF7>(</span><span class=cF0>CE1000DescriptorEntryTX</span><span class=cF7>)</span><span class=cF0> * E1000_TX_BUFF_COUNT,
|
||||
<a name="l271"></a> </span><span class=cFE>16</span><span class=cF0>,
|
||||
<a name="l272"></a> </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l273"></a>
|
||||
<a name="l274"></a> e1000.tx_de_buffer = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.tx_de_buffer_phys;
|
||||
<a name="l275"></a>
|
||||
<a name="l276"></a> e1000.tx_buffer_addr_phys = </span><span class=cF5>CAlloc</span><span class=cF0>(ETHERNET_FRAME_SIZE * E1000_TX_BUFF_COUNT, </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l233"></a> </span><span class=cF1>return</span><span class=cF0> de_index;
|
||||
<a name="l234"></a>}
|
||||
<a name="l235"></a>
|
||||
<a name="l236"></a></span><span class=cF1>U0</span><span class=cF0> E1000ReceivePacketRelease(</span><span class=cF9>I64</span><span class=cF0> de_index)
|
||||
<a name="l237"></a>{
|
||||
<a name="l238"></a> CE1000DescriptorEntryRX *entry = &e1000.rx_de_buffer[de_index * </span><span class=cF1>sizeof</span><span class=cF0>(CE1000DescriptorEntryRX)];
|
||||
<a name="l239"></a> entry->status = </span><span class=cFE>0</span><span class=cF0>;
|
||||
<a name="l240"></a>
|
||||
<a name="l241"></a> E1000MMIOWrite(E1000_REG_RDT, e1000.current_rx_de_index);
|
||||
<a name="l242"></a>}
|
||||
<a name="l243"></a>
|
||||
<a name="l244"></a></span><span class=cF1>Bool</span><span class=cF0> E1000DriverOwnsRX(CE1000DescriptorEntryRX *entry)
|
||||
<a name="l245"></a>{
|
||||
<a name="l246"></a> </span><span class=cF1>return</span><span class=cF0> </span><span class=cF5>Bt</span><span class=cF0>(&entry->status, </span><span class=cFE>0</span><span class=cF0>); </span><span class=cF2>// ?? TODO #define</span><span class=cF0>
|
||||
<a name="l247"></a>}
|
||||
<a name="l248"></a>
|
||||
<a name="l249"></a></span><span class=cF9>I64</span><span class=cF0> E1000TransmitPacketAllocate(</span><span class=cF1>U8</span><span class=cF0> **packet_buffer_out, </span><span class=cF9>I64</span><span class=cF0> length)
|
||||
<a name="l250"></a>{ </span><span class=cF9>I64</span><span class=cF0> de_index = e1000.current_tx_de_index;
|
||||
<a name="l251"></a>
|
||||
<a name="l252"></a> CE1000DescriptorEntryTX *entry = &e1000.tx_de_buffer[de_index * </span><span class=cF1>sizeof</span><span class=cF0>(CE1000DescriptorEntryTX)];
|
||||
<a name="l253"></a>
|
||||
<a name="l254"></a> *packet_buffer_out = e1000.tx_buffer_addr + de_index * ETHERNET_FRAME_SIZE;
|
||||
<a name="l255"></a>
|
||||
<a name="l256"></a> </span><span class=cF5>MemSet</span><span class=cF0>(*packet_buffer_out, </span><span class=cFE>0</span><span class=cF0>, ETHERNET_FRAME_SIZE); </span><span class=cF2>// Clear buffer contents in advance.</span><span class=cF0>
|
||||
<a name="l257"></a>
|
||||
<a name="l258"></a> entry->address = *packet_buffer_out;
|
||||
<a name="l259"></a> entry->length = length;
|
||||
<a name="l260"></a> entry->cmd = (</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>3</span><span class=cF0>) | </span><span class=cFE>3</span><span class=cF0>; </span><span class=cF2>// ??? TODO #define</span><span class=cF0>
|
||||
<a name="l261"></a>
|
||||
<a name="l262"></a> NetLog(</span><span class=cF6>"E1000 ALLOCATE TX PACKET: de_index: %X."</span><span class=cF0>, de_index);
|
||||
<a name="l263"></a> </span><span class=cF1>return</span><span class=cF0> de_index;
|
||||
<a name="l264"></a>}
|
||||
<a name="l265"></a>
|
||||
<a name="l266"></a></span><span class=cF1>U0</span><span class=cF0> E1000TransmitPacketFinish(</span><span class=cF9>I64</span><span class=cF0> de_index)
|
||||
<a name="l267"></a>{
|
||||
<a name="l268"></a> CE1000DescriptorEntryTX *entry = &e1000.tx_de_buffer[de_index * </span><span class=cF1>sizeof</span><span class=cF0>(CE1000DescriptorEntryTX)];
|
||||
<a name="l269"></a>
|
||||
<a name="l270"></a> e1000.current_tx_de_index = (e1000.current_tx_de_index + </span><span class=cFE>1</span><span class=cF0>) & (E1000_TX_BUFF_COUNT - </span><span class=cFE>1</span><span class=cF0>);
|
||||
<a name="l271"></a> E1000MMIOWrite(E1000_REG_TDT, e1000.current_tx_de_index);
|
||||
<a name="l272"></a>
|
||||
<a name="l273"></a> </span><span class=cF5>ClassRep</span><span class=cF0>(entry);
|
||||
<a name="l274"></a>
|
||||
<a name="l275"></a> </span><span class=cF1>while</span><span class=cF0> (!</span><span class=cF7>(</span><span class=cF0>entry->sta & </span><span class=cFE>0xF</span><span class=cF7>)</span><span class=cF0>) </span><span class=cF2>// ???</span><span class=cF0>
|
||||
<a name="l276"></a> </span><span class=cF5>Yield</span><span class=cF0>;
|
||||
<a name="l277"></a>
|
||||
<a name="l278"></a> e1000.tx_buffer_addr = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.tx_buffer_addr_phys;
|
||||
<a name="l279"></a>
|
||||
<a name="l280"></a> </span><span class=cF2>// setup tx de ring buffer</span><span class=cF0>
|
||||
<a name="l281"></a> E1000MMIOWrite(E1000_REG_TDBAH, e1000.tx_de_buffer >> </span><span class=cFE>32</span><span class=cF0>); </span><span class=cF2>// should we be using uncached addr here ?</span><span class=cF0>
|
||||
<a name="l282"></a> E1000MMIOWrite(E1000_REG_TDBAL, e1000.tx_de_buffer & </span><span class=cFE>0xFFFFFFFF</span><span class=cF0>);
|
||||
<a name="l283"></a>
|
||||
<a name="l284"></a> </span><span class=cF2>// set tx buffer length</span><span class=cF0>
|
||||
<a name="l285"></a> E1000MMIOWrite(E1000_REG_TDLEN, E1000_TX_BUFF_COUNT * </span><span class=cFE>16</span><span class=cF0>);
|
||||
<a name="l286"></a>
|
||||
<a name="l287"></a> </span><span class=cF2>// set head tail pointers</span><span class=cF0>
|
||||
<a name="l288"></a> E1000MMIOWrite(E1000_REG_TDH, </span><span class=cFE>0</span><span class=cF0>);
|
||||
<a name="l289"></a> E1000MMIOWrite(E1000_REG_TDT, E1000_RX_BUFF_COUNT);
|
||||
<a name="l290"></a>
|
||||
<a name="l291"></a> </span><span class=cF2>// set transmit control reg</span><span class=cF0>
|
||||
<a name="l292"></a> E1000MMIOWrite(E1000_REG_TCTL, E1000_TCTLF_EN | E1000_TCTLF_PSP);
|
||||
<a name="l293"></a>
|
||||
<a name="l294"></a>}
|
||||
<a name="l295"></a>
|
||||
<a name="l278"></a> NetLog(</span><span class=cF6>"E1000 FINISH TX PACKET: TX DE index: %X."</span><span class=cF0>, de_index);
|
||||
<a name="l279"></a>}
|
||||
<a name="l280"></a>
|
||||
<a name="l281"></a></span><span class=cF1>U0</span><span class=cF0> EthernetFrameFinish(</span><span class=cF9>I64</span><span class=cF0> de_index)
|
||||
<a name="l282"></a>{</span><span class=cF2>//Alias for driver Finish TX function.</span><span class=cF0>
|
||||
<a name="l283"></a> E1000TransmitPacketFinish(de_index);
|
||||
<a name="l284"></a>}
|
||||
<a name="l285"></a>
|
||||
<a name="l286"></a></span><span class=cF1>interrupt</span><span class=cF0> </span><span class=cF1>U0</span><span class=cF0> E1000IRQ()
|
||||
<a name="l287"></a>{ </span><span class=cF2>// TODO need #defines</span><span class=cF0>
|
||||
<a name="l288"></a> </span><span class=cF9>U32</span><span class=cF0> icr = E1000MMIORead(</span><span class=cFE>0xC0</span><span class=cF0>); </span><span class=cF2>// ???</span><span class=cF0>
|
||||
<a name="l289"></a> </span><span class=cF1>Bool</span><span class=cF0> poll = </span><span class=cF3>FALSE</span><span class=cF0>;
|
||||
<a name="l290"></a> CE1000DescriptorEntryRX *entry = e1000.rx_de_buffer;
|
||||
<a name="l291"></a> </span><span class=cF1>U8</span><span class=cF0> *packet_buffer;
|
||||
<a name="l292"></a> </span><span class=cF9>U16</span><span class=cF0> packet_length;
|
||||
<a name="l293"></a> </span><span class=cF9>I64</span><span class=cF0> de_index;
|
||||
<a name="l294"></a>
|
||||
<a name="l295"></a> icr &= ~</span><span class=cFE>3</span><span class=cF0>; </span><span class=cF2>// ??</span><span class=cF0>
|
||||
<a name="l296"></a>
|
||||
<a name="l297"></a></span><span class=cF1>U0</span><span class=cF0> E1000Init()
|
||||
<a name="l298"></a>{
|
||||
<a name="l299"></a> </span><span class=cF9>I64</span><span class=cF0> i;
|
||||
<a name="l300"></a>
|
||||
<a name="l301"></a> </span><span class=cF5>MemSet</span><span class=cF0>(&e1000, </span><span class=cFE>0</span><span class=cF0>, </span><span class=cF1>sizeof</span><span class=cF7>(</span><span class=cF0>CE1000</span><span class=cF7>)</span><span class=cF0>); </span><span class=cF2>// e1000 global var will hold member data the driver uses often.</span><span class=cF0>
|
||||
<a name="l302"></a> </span><span class=cF6>"\nE1000 driver WIP\n\n"</span><span class=cF0>;
|
||||
<a name="l303"></a>
|
||||
<a name="l304"></a> e1000.pci = E1000PCIDevFind;
|
||||
<a name="l305"></a> </span><span class=cF1>if</span><span class=cF0> (!e1000.pci)
|
||||
<a name="l306"></a> </span><span class=cF1>return</span><span class=cF0>; </span><span class=cF2>// if we don't find the card, quit.</span><span class=cF0>
|
||||
<a name="l307"></a>
|
||||
<a name="l308"></a>
|
||||
<a name="l309"></a> e1000.mmio_address = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.pci->base[</span><span class=cFE>0</span><span class=cF0>] & ~</span><span class=cFE>0xF</span><span class=cF0>;
|
||||
<a name="l310"></a> </span><span class=cF2>// Assuming card supports MMIO... lower 4 bits are hardwired zero (?)</span><span class=cF0>
|
||||
<a name="l311"></a>
|
||||
<a name="l312"></a> </span><span class=cF6>"\nMMIO address: 0x%0X\n"</span><span class=cF0>, e1000.mmio_address;
|
||||
<a name="l297"></a> </span><span class=cF1>if</span><span class=cF0> (icr & </span><span class=cF7>(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>2</span><span class=cF7>)</span><span class=cF0>) </span><span class=cF2>// ?? 'link status change' ??</span><span class=cF0>
|
||||
<a name="l298"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l299"></a> icr &= ~(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>2</span><span class=cF0>);
|
||||
<a name="l300"></a> E1000MMIOWrite(E1000_REG_CTRL, E1000MMIORead</span><span class=cF7>(</span><span class=cF0>E1000_REG_CTRL</span><span class=cF7>)</span><span class=cF0> | E1000_CTRLF_SLU);
|
||||
<a name="l301"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l302"></a>
|
||||
<a name="l303"></a> </span><span class=cF1>if</span><span class=cF0> (icr & </span><span class=cF7>(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>6</span><span class=cF7>)</span><span class=cF0> || icr & </span><span class=cF7>(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>4</span><span class=cF7>)</span><span class=cF0>) </span><span class=cF2>// ?? 'rx underrun / min threshold' ??</span><span class=cF0>
|
||||
<a name="l304"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l305"></a> icr = ~(</span><span class=cF7>(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>6</span><span class=cF7>)</span><span class=cF0> | </span><span class=cF7>(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>4</span><span class=cF7>)</span><span class=cF0>); </span><span class=cF2>// ??</span><span class=cF0>
|
||||
<a name="l306"></a>
|
||||
<a name="l307"></a> poll = </span><span class=cF3>TRUE</span><span class=cF0>;
|
||||
<a name="l308"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l309"></a>
|
||||
<a name="l310"></a> </span><span class=cF1>if</span><span class=cF0> (icr & </span><span class=cF7>(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>7</span><span class=cF7>)</span><span class=cF0>) </span><span class=cF2>// ?? 'packet pending?</span><span class=cF0>
|
||||
<a name="l311"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l312"></a> icr &= ~(</span><span class=cFE>1</span><span class=cF0> << </span><span class=cFE>7</span><span class=cF0>);
|
||||
<a name="l313"></a>
|
||||
<a name="l314"></a> </span><span class=cF2>// init rx/tx addrs? (linux)</span><span class=cF0>
|
||||
<a name="l315"></a>
|
||||
<a name="l316"></a> </span><span class=cF2>// eeprom? MAC ?</span><span class=cF0>
|
||||
<a name="l317"></a> E1000MACGet;
|
||||
<a name="l318"></a>
|
||||
<a name="l319"></a> </span><span class=cF2>// setup link? (01000101's driver)</span><span class=cF0>
|
||||
<a name="l320"></a> E1000MMIOWrite(E1000_REG_CTRL, E1000MMIORead</span><span class=cF7>(</span><span class=cF0>E1000_REG_CTRL</span><span class=cF7>)</span><span class=cF0> | E1000_CTRLF_SLU);
|
||||
<a name="l321"></a>
|
||||
<a name="l322"></a> </span><span class=cF2>// zero out multicast hash? (linux)</span><span class=cF0>
|
||||
<a name="l323"></a> </span><span class=cF2>// zero out multicast table array (01000101's driver)</span><span class=cF0>
|
||||
<a name="l324"></a> </span><span class=cF1>for</span><span class=cF0> (i = </span><span class=cFE>0</span><span class=cF0>; i < </span><span class=cFE>128</span><span class=cF0>; i++)
|
||||
<a name="l325"></a> E1000MMIOWrite(E1000_REG_MTA + i*</span><span class=cFE>4</span><span class=cF0>, </span><span class=cFE>0</span><span class=cF0>);
|
||||
<a name="l326"></a>
|
||||
<a name="l327"></a> </span><span class=cF2>// setup link? (linux)</span><span class=cF0>
|
||||
<a name="l328"></a>
|
||||
<a name="l329"></a> </span><span class=cF2>// clear all statistics regs after link establish attempt (linux)</span><span class=cF0>
|
||||
<a name="l330"></a>
|
||||
<a name="l331"></a> </span><span class=cF2>// enable & clear existing interupts (01000101's driver)</span><span class=cF0>
|
||||
<a name="l332"></a> E1000MMIOWrite(E1000_REG_IMS, </span><span class=cFE>0x1F6DC</span><span class=cF0>); </span><span class=cF2>// TODO: WHAT IS THIS</span><span class=cF0>
|
||||
<a name="l333"></a> E1000MMIORead(</span><span class=cFE>0xC0</span><span class=cF0>); </span><span class=cF2>// TODO : WHAT IS THIS ???</span><span class=cF0>
|
||||
<a name="l334"></a>
|
||||
<a name="l335"></a> </span><span class=cF2>// start rx tx?</span><span class=cF0>
|
||||
<a name="l336"></a> E1000RXInit;
|
||||
<a name="l337"></a> E1000TXInit;
|
||||
<a name="l338"></a>
|
||||
<a name="l339"></a>
|
||||
<a name="l340"></a> NetErr(</span><span class=cF6>"TODO E1000"</span><span class=cF0>);
|
||||
<a name="l341"></a> </span><span class=cF6>"\n"</span><span class=cF0>;
|
||||
<a name="l342"></a> </span><span class=cF5>ClassRep</span><span class=cF0>(&e1000);
|
||||
<a name="l343"></a>}
|
||||
<a name="l314"></a> poll = </span><span class=cF3>TRUE</span><span class=cF0>;
|
||||
<a name="l315"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l316"></a>
|
||||
<a name="l317"></a> </span><span class=cF1>if</span><span class=cF0> (poll)
|
||||
<a name="l318"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l319"></a> </span><span class=cF1>while</span><span class=cF0> (E1000DriverOwnsRX</span><span class=cF7>(</span><span class=cF0>&entry[e1000.current_rx_de_index]</span><span class=cF7>)</span><span class=cF0>)
|
||||
<a name="l320"></a> {
|
||||
<a name="l321"></a> NetLog(</span><span class=cF6>"$BG,LTCYAN$$FG,WHITE$"</span><span class=cF0>
|
||||
<a name="l322"></a> </span><span class=cF6>"==== E1000 IRQ ===="</span><span class=cF0>
|
||||
<a name="l323"></a> </span><span class=cF6>"$BG$$FG$"</span><span class=cF0>);
|
||||
<a name="l324"></a>
|
||||
<a name="l325"></a> NetLog(</span><span class=cF6>"$BD,CYAN$$FD,WHITE$"</span><span class=cF0>
|
||||
<a name="l326"></a> </span><span class=cF6>"E1000 IRQ: Saw owned RX DE index %d."</span><span class=cF0>, e1000.current_rx_de_index);
|
||||
<a name="l327"></a>
|
||||
<a name="l328"></a> de_index = E1000PacketReceive(&packet_buffer, &packet_length);
|
||||
<a name="l329"></a>
|
||||
<a name="l330"></a> </span><span class=cF1>if</span><span class=cF0> (de_index >= </span><span class=cFE>0</span><span class=cF0>)
|
||||
<a name="l331"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l332"></a> NetLog(</span><span class=cF6>"E1000 IRQ: Pushing copy into Net Queue, releasing receive packet."</span><span class=cF0>);
|
||||
<a name="l333"></a> NetQueuePush(packet_buffer, packet_length);
|
||||
<a name="l334"></a> E1000ReceivePacketRelease(de_index);
|
||||
<a name="l335"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l336"></a>
|
||||
<a name="l337"></a> NetLog(</span><span class=cF6>"E1000 IRQ: Exiting.\n"</span><span class=cF0>
|
||||
<a name="l338"></a> </span><span class=cF6>"$BD,WHITE$$FD,LTGRAY$"</span><span class=cF0>
|
||||
<a name="l339"></a> </span><span class=cF6>"$BG,LTCYAN$$FG,WHITE$"</span><span class=cF0>
|
||||
<a name="l340"></a> </span><span class=cF6>"==================="</span><span class=cF0>
|
||||
<a name="l341"></a> </span><span class=cF6>"$BG$$FG$"</span><span class=cF0>);
|
||||
<a name="l342"></a> }
|
||||
<a name="l343"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l344"></a>
|
||||
<a name="l345"></a></span><span class=cF9>I64</span><span class=cF0> EthernetFrameAllocate(</span><span class=cF1>U8</span><span class=cF0> **packet_buffer_out,
|
||||
<a name="l346"></a> </span><span class=cF1>U8</span><span class=cF0> *source_address,
|
||||
<a name="l347"></a> </span><span class=cF1>U8</span><span class=cF0> *destination_address,
|
||||
<a name="l348"></a> </span><span class=cF9>U16</span><span class=cF0> ethertype,
|
||||
<a name="l349"></a> </span><span class=cF9>I64</span><span class=cF0> packet_length)
|
||||
<a name="l350"></a>{
|
||||
<a name="l351"></a> </span><span class=cF1>no_warn</span><span class=cF0> packet_buffer_out, source_address, destination_address, ethertype, packet_length;
|
||||
<a name="l352"></a> NetErr(</span><span class=cF6>"TODO E1000"</span><span class=cF0>);
|
||||
<a name="l353"></a> </span><span class=cF1>return</span><span class=cF0> -</span><span class=cFE>1</span><span class=cF0>;
|
||||
<a name="l354"></a>}
|
||||
<a name="l355"></a>
|
||||
<a name="l356"></a></span><span class=cF1>U8</span><span class=cF0> *EthernetMACGet()
|
||||
<a name="l357"></a>{
|
||||
<a name="l358"></a> </span><span class=cF1>return</span><span class=cF0> e1000.mac_address;
|
||||
<a name="l359"></a>}
|
||||
<a name="l360"></a>
|
||||
<a name="l361"></a></span><span class=cF1>U0</span><span class=cF0> NetStop()
|
||||
<a name="l362"></a>{
|
||||
<a name="l363"></a>
|
||||
<a name="l364"></a>}
|
||||
<a name="l365"></a>
|
||||
<a name="l366"></a></span><span class=cF1>U0</span><span class=cF0> NetStart()
|
||||
<a name="l367"></a>{
|
||||
<a name="l345"></a>
|
||||
<a name="l346"></a>
|
||||
<a name="l347"></a> E1000MMIORead(</span><span class=cFE>0xC0</span><span class=cF0>); </span><span class=cF2>// ?? 'clear pending interrupts' ??</span><span class=cF0>
|
||||
<a name="l348"></a>
|
||||
<a name="l349"></a> *(</span><span class=cFB>dev</span><span class=cF0>.uncached_alias + </span><span class=cF3>LAPIC_EOI</span><span class=cF0>)(</span><span class=cF9>U32</span><span class=cF0>*) = </span><span class=cFE>0</span><span class=cF0>;
|
||||
<a name="l350"></a>
|
||||
<a name="l351"></a>}
|
||||
<a name="l352"></a>
|
||||
<a name="l353"></a></span><span class=cF1>U0</span><span class=cF0> PCIInterruptsReroute(</span><span class=cF9>I64</span><span class=cF0> base)
|
||||
<a name="l354"></a>{ </span><span class=cF2>// todo: comments explaining process, maybe better var names</span><span class=cF0>
|
||||
<a name="l355"></a> </span><span class=cF9>I64</span><span class=cF0> i;
|
||||
<a name="l356"></a> </span><span class=cF1>U8</span><span class=cF0> *da = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + </span><span class=cF3>IOAPIC_REG</span><span class=cF0>;
|
||||
<a name="l357"></a> </span><span class=cF9>U32</span><span class=cF0> *_d = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + </span><span class=cF3>IOAPIC_DATA</span><span class=cF0>;
|
||||
<a name="l358"></a>
|
||||
<a name="l359"></a> </span><span class=cF1>for</span><span class=cF0> (i = </span><span class=cFE>0</span><span class=cF0>; i < </span><span class=cFE>4</span><span class=cF0>; i++)
|
||||
<a name="l360"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l361"></a> *da = </span><span class=cF3>IOREDTAB</span><span class=cF0> + i * </span><span class=cFE>2</span><span class=cF0> + </span><span class=cFE>1</span><span class=cF0>;
|
||||
<a name="l362"></a> *_d = </span><span class=cFB>dev</span><span class=cF0>.mp_apic_ids[INT_DEST_CPU] << </span><span class=cFE>24</span><span class=cF0>;
|
||||
<a name="l363"></a> *da = </span><span class=cF3>IOREDTAB</span><span class=cF0> + i * </span><span class=cFE>2</span><span class=cF0>;
|
||||
<a name="l364"></a> *_d = </span><span class=cFE>0x4000</span><span class=cF0> + base + i;
|
||||
<a name="l365"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l366"></a>}
|
||||
<a name="l367"></a>
|
||||
<a name="l368"></a>
|
||||
<a name="l369"></a>}
|
||||
<a name="l370"></a>
|
||||
<a name="l371"></a>E1000Init;</span></pre></body>
|
||||
<a name="l369"></a></span><span class=cF1>U0</span><span class=cF0> E1000InterruptsSetup()
|
||||
<a name="l370"></a>{ </span><span class=cF2>// .. ?</span><span class=cF0>
|
||||
<a name="l371"></a> </span><span class=cF5>IntEntrySet</span><span class=cF0>(I_E1000_0, &E1000IRQ);
|
||||
<a name="l372"></a> </span><span class=cF5>IntEntrySet</span><span class=cF0>(I_E1000_1, &E1000IRQ);
|
||||
<a name="l373"></a> </span><span class=cF5>IntEntrySet</span><span class=cF0>(I_E1000_2, &E1000IRQ);
|
||||
<a name="l374"></a> </span><span class=cF5>IntEntrySet</span><span class=cF0>(I_E1000_3, &E1000IRQ);
|
||||
<a name="l375"></a> PCIInterruptsReroute(I_E1000_0);
|
||||
<a name="l376"></a>}
|
||||
<a name="l377"></a>
|
||||
<a name="l378"></a></span><span class=cF1>U0</span><span class=cF0> E1000InitRX()
|
||||
<a name="l379"></a>{
|
||||
<a name="l380"></a> </span><span class=cF9>I64</span><span class=cF0> de_index;
|
||||
<a name="l381"></a>
|
||||
<a name="l382"></a> e1000.rx_de_buffer_phys = </span><span class=cF5>CAllocAligned</span><span class=cF0>(</span><span class=cF1>sizeof</span><span class=cF7>(</span><span class=cF0>CE1000DescriptorEntryRX</span><span class=cF7>)</span><span class=cF0> * E1000_RX_BUFF_COUNT,
|
||||
<a name="l383"></a> </span><span class=cFE>16</span><span class=cF0>,
|
||||
<a name="l384"></a> </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l385"></a>
|
||||
<a name="l386"></a> e1000.rx_de_buffer = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.rx_de_buffer_phys;
|
||||
<a name="l387"></a>
|
||||
<a name="l388"></a> e1000.rx_buffer_addr_phys = </span><span class=cF5>CAlloc</span><span class=cF0>(ETHERNET_FRAME_SIZE * E1000_RX_BUFF_COUNT, </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l389"></a>
|
||||
<a name="l390"></a> e1000.rx_buffer_addr = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.rx_buffer_addr_phys;
|
||||
<a name="l391"></a>
|
||||
<a name="l392"></a> </span><span class=cF2>// iterate de's and make packet buffers for each</span><span class=cF0>
|
||||
<a name="l393"></a> CE1000DescriptorEntryRX *entry = e1000.rx_de_buffer;
|
||||
<a name="l394"></a> </span><span class=cF1>for</span><span class=cF0> (de_index = </span><span class=cFE>0</span><span class=cF0>; de_index < E1000_RX_BUFF_COUNT; de_index++)
|
||||
<a name="l395"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l396"></a> entry->address = e1000.rx_buffer_addr + de_index * ETHERNET_FRAME_SIZE; </span><span class=cF2>// is this right? might need to change ?..</span><span class=cF0>
|
||||
<a name="l397"></a> </span><span class=cF2>// 01000101 MAlloc's 8208 for each DE</span><span class=cF0>
|
||||
<a name="l398"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>PCNetDescriptorEntryInit(&entry[de_index],</span><span class=cF0>
|
||||
<a name="l399"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2> pcnet.rx_buffer_addr + de_index * ETHERNET_FRAME_SIZE,</span><span class=cF0>
|
||||
<a name="l400"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2> TRUE); // TRUE for is_rx.</span><span class=cF0>
|
||||
<a name="l401"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l402"></a>
|
||||
<a name="l403"></a> </span><span class=cF2>// setup rx de ring buffer</span><span class=cF0>
|
||||
<a name="l404"></a> E1000MMIOWrite(E1000_REG_RDBAH, e1000.rx_de_buffer >> </span><span class=cFE>32</span><span class=cF0>); </span><span class=cF2>// should we be using uncached addr here ?</span><span class=cF0>
|
||||
<a name="l405"></a> E1000MMIOWrite(E1000_REG_RDBAL, e1000.rx_de_buffer & </span><span class=cFE>0xFFFFFFFF</span><span class=cF0>);
|
||||
<a name="l406"></a>
|
||||
<a name="l407"></a> </span><span class=cF2>// set receive buffer length</span><span class=cF0>
|
||||
<a name="l408"></a> E1000MMIOWrite(E1000_REG_RDLEN, E1000_RX_BUFF_COUNT * </span><span class=cFE>16</span><span class=cF0>);
|
||||
<a name="l409"></a>
|
||||
<a name="l410"></a> </span><span class=cF2>// set head tail pointers</span><span class=cF0>
|
||||
<a name="l411"></a> E1000MMIOWrite(E1000_REG_RDH, </span><span class=cFE>0</span><span class=cF0>);
|
||||
<a name="l412"></a> E1000MMIOWrite(E1000_REG_RDT, E1000_RX_BUFF_COUNT);
|
||||
<a name="l413"></a>
|
||||
<a name="l414"></a> </span><span class=cF2>// set receive control reg</span><span class=cF0>
|
||||
<a name="l415"></a> E1000MMIOWrite(E1000_REG_RCTL, E1000_RCTLF_SBP |
|
||||
<a name="l416"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_UPE</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l417"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_RDMTS_HALF</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l418"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_BSIZE_2048</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l419"></a></span><span class=cF2>//</span><span class=cF0> </span><span class=cF2>E1000_RCTLF_MPE</span><span class=cF0> </span><span class=cF2>|</span><span class=cF0>
|
||||
<a name="l420"></a> E1000_RCTLF_SECRC |
|
||||
<a name="l421"></a> E1000_RCTLF_LPE |
|
||||
<a name="l422"></a> E1000_RCTLF_BAM);
|
||||
<a name="l423"></a>}
|
||||
<a name="l424"></a>
|
||||
<a name="l425"></a></span><span class=cF1>U0</span><span class=cF0> E1000InitTX()
|
||||
<a name="l426"></a>{
|
||||
<a name="l427"></a> e1000.tx_de_buffer_phys = </span><span class=cF5>CAllocAligned</span><span class=cF0>(</span><span class=cF1>sizeof</span><span class=cF7>(</span><span class=cF0>CE1000DescriptorEntryTX</span><span class=cF7>)</span><span class=cF0> * E1000_TX_BUFF_COUNT,
|
||||
<a name="l428"></a> </span><span class=cFE>16</span><span class=cF0>,
|
||||
<a name="l429"></a> </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l430"></a>
|
||||
<a name="l431"></a> e1000.tx_de_buffer = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.tx_de_buffer_phys;
|
||||
<a name="l432"></a>
|
||||
<a name="l433"></a> e1000.tx_buffer_addr_phys = </span><span class=cF5>CAlloc</span><span class=cF0>(ETHERNET_FRAME_SIZE * E1000_TX_BUFF_COUNT, </span><span class=cF5>Fs</span><span class=cF0>->code_heap);
|
||||
<a name="l434"></a>
|
||||
<a name="l435"></a> e1000.tx_buffer_addr = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.tx_buffer_addr_phys;
|
||||
<a name="l436"></a>
|
||||
<a name="l437"></a> </span><span class=cF2>// setup tx de ring buffer</span><span class=cF0>
|
||||
<a name="l438"></a> E1000MMIOWrite(E1000_REG_TDBAH, e1000.tx_de_buffer >> </span><span class=cFE>32</span><span class=cF0>); </span><span class=cF2>// should we be using uncached addr here ?</span><span class=cF0>
|
||||
<a name="l439"></a> E1000MMIOWrite(E1000_REG_TDBAL, e1000.tx_de_buffer & </span><span class=cFE>0xFFFFFFFF</span><span class=cF0>);
|
||||
<a name="l440"></a>
|
||||
<a name="l441"></a> </span><span class=cF2>// set tx buffer length</span><span class=cF0>
|
||||
<a name="l442"></a> E1000MMIOWrite(E1000_REG_TDLEN, E1000_TX_BUFF_COUNT * </span><span class=cFE>16</span><span class=cF0>);
|
||||
<a name="l443"></a>
|
||||
<a name="l444"></a> </span><span class=cF2>// set head tail pointers</span><span class=cF0>
|
||||
<a name="l445"></a> E1000MMIOWrite(E1000_REG_TDH, </span><span class=cFE>0</span><span class=cF0>);
|
||||
<a name="l446"></a> E1000MMIOWrite(E1000_REG_TDT, E1000_RX_BUFF_COUNT);
|
||||
<a name="l447"></a>
|
||||
<a name="l448"></a> </span><span class=cF2>// set transmit control reg</span><span class=cF0>
|
||||
<a name="l449"></a> E1000MMIOWrite(E1000_REG_TCTL, E1000_TCTLF_EN | E1000_TCTLF_PSP);
|
||||
<a name="l450"></a>
|
||||
<a name="l451"></a>}
|
||||
<a name="l452"></a>
|
||||
<a name="l453"></a>
|
||||
<a name="l454"></a></span><span class=cF1>U0</span><span class=cF0> E1000Init()
|
||||
<a name="l455"></a>{
|
||||
<a name="l456"></a> </span><span class=cF9>I64</span><span class=cF0> i;
|
||||
<a name="l457"></a>
|
||||
<a name="l458"></a> </span><span class=cF5>MemSet</span><span class=cF0>(&e1000, </span><span class=cFE>0</span><span class=cF0>, </span><span class=cF1>sizeof</span><span class=cF7>(</span><span class=cF0>CE1000</span><span class=cF7>)</span><span class=cF0>); </span><span class=cF2>// e1000 global var will hold member data the driver uses often.</span><span class=cF0>
|
||||
<a name="l459"></a> </span><span class=cF6>"\nE1000 driver WIP\n\n"</span><span class=cF0>;
|
||||
<a name="l460"></a>
|
||||
<a name="l461"></a> e1000.pci = E1000PCIDevFind;
|
||||
<a name="l462"></a> </span><span class=cF1>if</span><span class=cF0> (!e1000.pci)
|
||||
<a name="l463"></a> </span><span class=cF1>return</span><span class=cF0>; </span><span class=cF2>// if we don't find the card, quit.</span><span class=cF0>
|
||||
<a name="l464"></a>
|
||||
<a name="l465"></a>
|
||||
<a name="l466"></a> e1000.mmio_address = </span><span class=cFB>dev</span><span class=cF0>.uncached_alias + e1000.pci->base[</span><span class=cFE>0</span><span class=cF0>] & ~</span><span class=cFE>0xF</span><span class=cF0>;
|
||||
<a name="l467"></a> </span><span class=cF2>// Assuming card supports MMIO... lower 4 bits are hardwired zero (?)</span><span class=cF0>
|
||||
<a name="l468"></a>
|
||||
<a name="l469"></a> </span><span class=cF6>"\nMMIO address: 0x%0X\n"</span><span class=cF0>, e1000.mmio_address;
|
||||
<a name="l470"></a>
|
||||
<a name="l471"></a> </span><span class=cF2>// init rx/tx addrs? (linux)</span><span class=cF0>
|
||||
<a name="l472"></a>
|
||||
<a name="l473"></a> </span><span class=cF2>// eeprom? MAC ?</span><span class=cF0>
|
||||
<a name="l474"></a> E1000MACGet;
|
||||
<a name="l475"></a>
|
||||
<a name="l476"></a> </span><span class=cF2>// setup link? (01000101's driver)</span><span class=cF0>
|
||||
<a name="l477"></a> E1000MMIOWrite(E1000_REG_CTRL, E1000MMIORead</span><span class=cF7>(</span><span class=cF0>E1000_REG_CTRL</span><span class=cF7>)</span><span class=cF0> | E1000_CTRLF_SLU);
|
||||
<a name="l478"></a>
|
||||
<a name="l479"></a> </span><span class=cF2>// zero out multicast hash? (linux)</span><span class=cF0>
|
||||
<a name="l480"></a> </span><span class=cF2>// zero out multicast table array (01000101's driver)</span><span class=cF0>
|
||||
<a name="l481"></a> </span><span class=cF1>for</span><span class=cF0> (i = </span><span class=cFE>0</span><span class=cF0>; i < </span><span class=cFE>128</span><span class=cF0>; i++)
|
||||
<a name="l482"></a> E1000MMIOWrite(E1000_REG_MTA + i*</span><span class=cFE>4</span><span class=cF0>, </span><span class=cFE>0</span><span class=cF0>);
|
||||
<a name="l483"></a>
|
||||
<a name="l484"></a> </span><span class=cF2>// setup link? (linux)</span><span class=cF0>
|
||||
<a name="l485"></a>
|
||||
<a name="l486"></a> </span><span class=cF2>// clear all statistics regs after link establish attempt (linux)</span><span class=cF0>
|
||||
<a name="l487"></a>
|
||||
<a name="l488"></a> </span><span class=cF2>// enable & clear existing interupts (01000101's driver)</span><span class=cF0>
|
||||
<a name="l489"></a> E1000MMIOWrite(E1000_REG_IMS, </span><span class=cFE>0x1F6DC</span><span class=cF0>); </span><span class=cF2>// TODO: WHAT IS THIS</span><span class=cF0>
|
||||
<a name="l490"></a> E1000MMIORead(</span><span class=cFE>0xC0</span><span class=cF0>); </span><span class=cF2>// TODO : WHAT IS THIS ???</span><span class=cF0>
|
||||
<a name="l491"></a>
|
||||
<a name="l492"></a> </span><span class=cF2>// start rx tx?</span><span class=cF0>
|
||||
<a name="l493"></a> E1000InitRX;
|
||||
<a name="l494"></a> E1000InitTX;
|
||||
<a name="l495"></a>
|
||||
<a name="l496"></a> E1000InterruptsSetup;
|
||||
<a name="l497"></a>
|
||||
<a name="l498"></a>
|
||||
<a name="l499"></a> NetErr(</span><span class=cF6>"TODO E1000"</span><span class=cF0>);
|
||||
<a name="l500"></a> </span><span class=cF6>"\n"</span><span class=cF0>;
|
||||
<a name="l501"></a> </span><span class=cF5>ClassRep</span><span class=cF0>(&e1000);
|
||||
<a name="l502"></a>}
|
||||
<a name="l503"></a>
|
||||
<a name="l504"></a></span><span class=cF9>I64</span><span class=cF0> EthernetFrameAllocate(</span><span class=cF1>U8</span><span class=cF0> **packet_buffer_out,
|
||||
<a name="l505"></a> </span><span class=cF1>U8</span><span class=cF0> *source_address,
|
||||
<a name="l506"></a> </span><span class=cF1>U8</span><span class=cF0> *destination_address,
|
||||
<a name="l507"></a> </span><span class=cF9>U16</span><span class=cF0> ethertype,
|
||||
<a name="l508"></a> </span><span class=cF9>I64</span><span class=cF0> packet_length)
|
||||
<a name="l509"></a>{
|
||||
<a name="l510"></a> </span><span class=cF1>U8</span><span class=cF0> *ethernet_frame;
|
||||
<a name="l511"></a> </span><span class=cF9>I64</span><span class=cF0> de_index;
|
||||
<a name="l512"></a>
|
||||
<a name="l513"></a> </span><span class=cF2>//need to see if 3 years later VirtualBox supports APAD_XMT!</span><span class=cF0>
|
||||
<a name="l514"></a> </span><span class=cF1>if</span><span class=cF0> (packet_length < ETHERNET_MIN_FRAME_SIZE)
|
||||
<a name="l515"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l516"></a> NetWarn(</span><span class=cF6>"ETHERNET FRAME ALLOCATE: Truncating length"</span><span class=cF0>);
|
||||
<a name="l517"></a> packet_length = ETHERNET_MIN_FRAME_SIZE;
|
||||
<a name="l518"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l519"></a>
|
||||
<a name="l520"></a> de_index = E1000TransmitPacketAllocate(&ethernet_frame, ETHERNET_MAC_HEADER_LENGTH + packet_length);
|
||||
<a name="l521"></a>
|
||||
<a name="l522"></a> </span><span class=cF1>if</span><span class=cF0> (de_index < </span><span class=cFE>0</span><span class=cF0>)
|
||||
<a name="l523"></a> </span><span class=cF7>{</span><span class=cF0>
|
||||
<a name="l524"></a> NetErr(</span><span class=cF6>"ETHERNET FRAME ALLOCATE: Failure"</span><span class=cF0>);
|
||||
<a name="l525"></a> </span><span class=cF1>return</span><span class=cF0> -</span><span class=cFE>1</span><span class=cF0>; </span><span class=cF2>// Positive value expected. Functions calling this must factor this in.</span><span class=cF0>
|
||||
<a name="l526"></a> </span><span class=cF7>}</span><span class=cF0>
|
||||
<a name="l527"></a>
|
||||
<a name="l528"></a> </span><span class=cF5>MemCopy</span><span class=cF0>(ethernet_frame, destination_address, MAC_ADDRESS_LENGTH);
|
||||
<a name="l529"></a> </span><span class=cF5>MemCopy</span><span class=cF0>(ethernet_frame + MAC_ADDRESS_LENGTH, source_address, MAC_ADDRESS_LENGTH);
|
||||
<a name="l530"></a>
|
||||
<a name="l531"></a> ethernet_frame[ETHERNET_ETHERTYPE_OFFSET] = ethertype >> </span><span class=cFE>8</span><span class=cF0>;
|
||||
<a name="l532"></a> ethernet_frame[ETHERNET_ETHERTYPE_OFFSET + </span><span class=cFE>1</span><span class=cF0>] = ethertype & </span><span class=cFE>0xFF</span><span class=cF0>;
|
||||
<a name="l533"></a>
|
||||
<a name="l534"></a> *packet_buffer_out = ethernet_frame + ETHERNET_MAC_HEADER_LENGTH;
|
||||
<a name="l535"></a>
|
||||
<a name="l536"></a> </span><span class=cF1>return</span><span class=cF0> de_index;
|
||||
<a name="l537"></a>
|
||||
<a name="l538"></a>}
|
||||
<a name="l539"></a>
|
||||
<a name="l540"></a></span><span class=cF1>U8</span><span class=cF0> *EthernetMACGet()
|
||||
<a name="l541"></a>{
|
||||
<a name="l542"></a> </span><span class=cF1>return</span><span class=cF0> e1000.mac_address;
|
||||
<a name="l543"></a>}
|
||||
<a name="l544"></a>
|
||||
<a name="l545"></a></span><span class=cF1>U0</span><span class=cF0> NetStop()
|
||||
<a name="l546"></a>{
|
||||
<a name="l547"></a>
|
||||
<a name="l548"></a>}
|
||||
<a name="l549"></a>
|
||||
<a name="l550"></a></span><span class=cF1>U0</span><span class=cF0> NetStart()
|
||||
<a name="l551"></a>{
|
||||
<a name="l552"></a>
|
||||
<a name="l553"></a>}
|
||||
<a name="l554"></a>
|
||||
<a name="l555"></a>E1000Init;</span></pre></body>
|
||||
</html>
|
||||
|
|
|
@ -25,7 +25,7 @@ body {background-color:#000000;}
|
|||
</head>
|
||||
<body>
|
||||
<pre style="font-family:monospace;font-size:12pt">
|
||||
<a name="l1"></a><span class=cF5>Cd</span><span class=cF0>(</span><span class=cF3>__DIR__</span><span class=cF0>);
|
||||
<a name="l1"></a><span class=cF5>Cd</span><span class=cF0>(</span><span class=cF3>__DIR__</span><span class=cF0>);;
|
||||
<a name="l2"></a>#</span><span class=cF1>include</span><span class=cF0> </span><span class=cF6>"Utilities/Net.HH"</span><span class=cF0>
|
||||
<a name="l3"></a>
|
||||
<a name="l4"></a>#</span><span class=cF1>include</span><span class=cF0> </span><span class=cF6>"Utilities/NetLog"</span><span class=cF0>
|
||||
|
|
|
@ -58,8 +58,14 @@ body {background-color:#000000;}
|
|||
<a name="l31"></a>#</span><span class=cF1>define</span><span class=cF0> I_PCNET1 </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>1</span><span class=cF0>
|
||||
<a name="l32"></a>#</span><span class=cF1>define</span><span class=cF0> I_PCNET2 </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>2</span><span class=cF0>
|
||||
<a name="l33"></a>#</span><span class=cF1>define</span><span class=cF0> I_PCNET3 </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>3</span><span class=cF0>
|
||||
<a name="l34"></a>#</span><span class=cF1>define</span><span class=cF0> I_NETHANDLER </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>4</span><span class=cF0>
|
||||
<a name="l35"></a>
|
||||
<a name="l36"></a>#</span><span class=cF1>define</span><span class=cF0> INT_DEST_CPU </span><span class=cFE>0</span><span class=cF0>
|
||||
<a name="l34"></a>
|
||||
<a name="l35"></a>#</span><span class=cF1>define</span><span class=cF0> I_E1000_0 </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>0</span><span class=cF0>
|
||||
<a name="l36"></a>#</span><span class=cF1>define</span><span class=cF0> I_E1000_1 </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>1</span><span class=cF0>
|
||||
<a name="l37"></a>#</span><span class=cF1>define</span><span class=cF0> I_E1000_2 </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>2</span><span class=cF0>
|
||||
<a name="l38"></a>#</span><span class=cF1>define</span><span class=cF0> I_E1000_3 </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>3</span><span class=cF0>
|
||||
<a name="l39"></a>
|
||||
<a name="l40"></a>#</span><span class=cF1>define</span><span class=cF0> I_NETHANDLER </span><span class=cF3>I_USER</span><span class=cF0> + </span><span class=cFE>4</span><span class=cF0>
|
||||
<a name="l41"></a>
|
||||
<a name="l42"></a>#</span><span class=cF1>define</span><span class=cF0> INT_DEST_CPU </span><span class=cFE>0</span><span class=cF0>
|
||||
</span></pre></body>
|
||||
</html>
|
||||
|
|
|
@ -1296,7 +1296,7 @@ body {background-color:#000000;}
|
|||
<a name="l1268"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_MEM_MIN_MEG </span></a><span class=cF0>256Meg 0000 0000000008 DefineStr </span><span class=cF4>
|
||||
<a name="l1269"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_MP_VECT </span></a><span class=cF0>00097000 0000 0000000010 DefineStr </span><span class=cF4>
|
||||
<a name="l1270"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_MP_VECT_END </span></a><span class=cF0>00097030 0000 0000000010 DefineStr </span><span class=cF4>
|
||||
<a name="l1271"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_OS_NAME_VERSION </span></a><span class=cF0>ZealOS V0. 0142 0000000010 DefineStr </span><span class=cF4>
|
||||
<a name="l1271"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_OS_NAME_VERSION </span></a><span class=cF0>ZealOS V0. 0154 0000000010 DefineStr </span><span class=cF4>
|
||||
<a name="l1272"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_PHYSICAL_SPACE_END</span></a><span class=cF0>011EBFFFFF 0000 0000000010 DefineStr </span><span class=cF4>
|
||||
<a name="l1273"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_SYS_FIXED_AREA_BASE</span></a><span class=cF0>00100000 0000 0000000010 DefineStr </span><span class=cF4>
|
||||
<a name="l1274"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l108"><span class=cF4>DD_SYS_FIXED_AREA_END</span></a><span class=cF0>00101FFF 0000 0000000010 DefineStr </span><span class=cF4>
|
||||
|
@ -1686,7 +1686,7 @@ body {background-color:#000000;}
|
|||
<a name="l1658"></a></span><a href="https://tomawezome.github.io/ZealOS/System/BlkDev/FileMgr.CC.html#l2"><span class=cF4>DirFileDoc </span></a><span class=cF0>003794D028 0002 0000000108 Funct </span><span class=cF4>
|
||||
<a name="l1659"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/BlkDev/DiskDirA.CC.html#l152"><span class=cF4>DirFilesFlatten </span></a><span class=cF0>000002844B 0001 ExportSysSym Imm </span><span class=cF4>
|
||||
<a name="l1660"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/BlkDev/DiskDirA.CC.html#l99"><span class=cF4>DirFilesSort </span></a><span class=cF0>0000028313 0001 ExportSysSym Imm </span><span class=cF4>
|
||||
<a name="l1661"></a></span><a href="https://tomawezome.github.io/ZealOS/Demo/ToHtmlToTXTDemo/HtmlDirList.CC.html#l1"><span class=cF4>DirIndexList </span></a><span class=cF0>00378798D8 0001 0000000193 Funct </span><span class=cF4>
|
||||
<a name="l1661"></a></span><a href="https://tomawezome.github.io/ZealOS/Demo/ToHtmlToTXTDemo/HtmlDirList.CC.html#l1"><span class=cF4>DirIndexList </span></a><span class=cF0>00378834D8 0001 0000000193 Funct </span><span class=cF4>
|
||||
<a name="l1662"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/BlkDev/FileSysFAT.CC.html#l213"><span class=cF4>DirLongNameFill </span></a><span class=cF0>000002AD68 0001 ExportSysSym Imm </span><span class=cF4>
|
||||
<a name="l1663"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/BlkDev/DiskDirB.CC.html#l184"><span class=cF4>DirMake </span></a><span class=cF0>000002E500 000D 0000000171 Funct Public </span><span class=cF4>
|
||||
<a name="l1664"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/BlkDev/DiskStrA.CC.html#l118"><span class=cF4>DirNameAbs </span></a><span class=cF0>0000020FCA 0009 0000000304 Funct Public </span><span class=cF4>
|
||||
|
@ -4772,8 +4772,8 @@ body {background-color:#000000;}
|
|||
<a name="l4744"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l246"><span class=cF4>ST_COLORS </span></a><span class=cF0>BLACK 10 0001 0000000068 DefineStr </span><span class=cF4>
|
||||
<a name="l4745"></a></span><a href="https://tomawezome.github.io/ZealOS/System/Utils/SysRep.CC.html#l250"><span class=cF4>ST_CPUID_1_ECX_FLAGS</span></a><span class=cF0>SSE3 13B 0000 00000011D8 DefineStr </span><span class=cF4>
|
||||
<a name="l4746"></a></span><a href="https://tomawezome.github.io/ZealOS/System/Utils/SysRep.CC.html#l215"><span class=cF4>ST_CPUID_1_EDX_FLAGS</span></a><span class=cF0>x87 FPU 20 0000 00000001D0 DefineStr </span><span class=cF4>
|
||||
<a name="l4747"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l159"><span class=cF4>ST_DAYS_OF_WEEK </span></a><span class=cF0>Sunday 07 0005 0000000040 DefineStr </span><span class=cF4>
|
||||
<a name="l4748"></a></span><a href="https://tomawezome.github.io/ZealOS/System/DolDoc/DocInit.CC.html#l14"><span class=cF4>ST_DOC_CMDS </span></a><span class=cF0>TX 2B 018C 0000000088 DefineStr </span><span class=cF4>
|
||||
<a name="l4747"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l159"><span class=cF4>ST_DAYS_OF_WEEK </span></a><span class=cF0>Sunday 07 0006 0000000040 DefineStr </span><span class=cF4>
|
||||
<a name="l4748"></a></span><a href="https://tomawezome.github.io/ZealOS/System/DolDoc/DocInit.CC.html#l14"><span class=cF4>ST_DOC_CMDS </span></a><span class=cF0>TX 2B 019F 0000000088 DefineStr </span><span class=cF4>
|
||||
<a name="l4749"></a></span><a href="https://tomawezome.github.io/ZealOS/System/DolDoc/DocInit.CC.html#l20"><span class=cF4>ST_DOC_FLAGS </span></a><span class=cF0>T 3E 0007 00000000C0 DefineStr </span><span class=cF4>
|
||||
<a name="l4750"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KDefine.CC.html#l228"><span class=cF4>ST_DRIVE_TYPES </span></a><span class=cF0>NULL 06 0002 0000000028 DefineStr </span><span class=cF4>
|
||||
<a name="l4751"></a></span><a href="https://tomawezome.github.io/ZealOS/Kernel/KernelA.HH.html#l4125"><span class=cF4>ST_ERR_ST </span></a><span class=cF0>\"\d\dRED\ 0003 0000000028 DefineStr </span><span class=cF4>
|
||||
|
|
|
@ -109,15 +109,21 @@ class CE1000
|
|||
|
||||
U8 mac_address[6];
|
||||
U64 mmio_address;
|
||||
|
||||
I64 current_rx_de_index; // Current Receive DE being processed. Gets incremented, wrapped to 0 at max of E1000_RX_BUFF_COUNT.
|
||||
I64 current_tx_de_index; // Current Transmit DE being processed. Gets incremented, wrapped to 0 at max of E1000_TX_BUFF_COUNT.
|
||||
|
||||
U8 *rx_de_buffer; // Uncached-alias of pointer to the buffer of RX Descriptor Entries.
|
||||
U8 *tx_de_buffer; // Uncached-alias of pointer to the buffer of TX Descriptor Entries.
|
||||
U8 *rx_de_buffer_phys; // Pointer to the buffer of RX Descriptor Entries. (Code Heap, lower 2Gb)
|
||||
U8 *tx_de_buffer_phys; // Pointer to the buffer of TX Descriptor Entries. (Code Heap, lower 2Gb)
|
||||
|
||||
U64 rx_buffer_addr; // Uncached-alias of address of receive buffers.
|
||||
U64 tx_buffer_addr; // Uncached-alias of address of transmit buffers.
|
||||
U64 rx_buffer_addr_phys; // Physical address of actual receive buffers (< 4 Gb)
|
||||
U64 tx_buffer_addr_phys; // Physical address of actual transmit buffers (< 4 Gb)
|
||||
|
||||
|
||||
} e1000; // e1000 is the global variable we store all of this into.
|
||||
|
||||
CPCIDev *E1000PCIDevFind()
|
||||
|
@ -175,7 +181,7 @@ U0 E1000MACGet()
|
|||
I64 i;
|
||||
U16 mac;
|
||||
|
||||
NetLog("E1000 GET MAC: Getting VM MAC.");
|
||||
NetLog("E1000 MAC GET: Getting MAC Address.");
|
||||
|
||||
for (i = 0; i < 3; i++)
|
||||
{
|
||||
|
@ -187,14 +193,163 @@ U0 E1000MACGet()
|
|||
|
||||
}
|
||||
|
||||
U0 EthernetFrameFinish(I64 de_index)
|
||||
{//Alias for driver Finish TX function.
|
||||
//E1000TransmitPacketFinish(de_index);
|
||||
no_warn de_index;
|
||||
NetErr("TODO E1000");
|
||||
I64 E1000PacketReceive(U8 **packet_buffer_out, U16 *packet_length_out)
|
||||
{
|
||||
I64 de_index = e1000.current_rx_de_index;
|
||||
CE1000DescriptorEntryRX *entry = &e1000.rx_de_buffer[de_index * sizeof(CE1000DescriptorEntryRX)];
|
||||
Bool drop = FALSE;
|
||||
|
||||
if (entry->length < 60)
|
||||
{
|
||||
NetErr("E1000 PACKET RECEIVE: Short Packet");
|
||||
drop = TRUE;
|
||||
}
|
||||
|
||||
if (entry->status & (1 << 1)) // EOP ?? TODO #define
|
||||
{
|
||||
NetErr("E1000 PACKET RECEIVE: No EOP Set");
|
||||
drop = TRUE;
|
||||
}
|
||||
|
||||
if (entry->errors)
|
||||
{
|
||||
NetErr("E1000 PACKET RECEIVE: RX DE Error Bits Set");
|
||||
drop = TRUE;
|
||||
}
|
||||
|
||||
e1000.current_rx_de_index = (e1000.current_rx_de_index + 1) & (E1000_RX_BUFF_COUNT - 1);
|
||||
|
||||
if (!drop)
|
||||
{
|
||||
*packet_buffer_out = entry->address;
|
||||
*packet_length_out = entry->length;
|
||||
}
|
||||
else
|
||||
{
|
||||
NetErr("E1000 PACKET RECEIVE: Dropping packet.");
|
||||
de_index = -1;
|
||||
}
|
||||
|
||||
return de_index;
|
||||
}
|
||||
|
||||
U0 E1000ReceivePacketRelease(I64 de_index)
|
||||
{
|
||||
CE1000DescriptorEntryRX *entry = &e1000.rx_de_buffer[de_index * sizeof(CE1000DescriptorEntryRX)];
|
||||
entry->status = 0;
|
||||
|
||||
E1000MMIOWrite(E1000_REG_RDT, e1000.current_rx_de_index);
|
||||
}
|
||||
|
||||
Bool E1000DriverOwnsRX(CE1000DescriptorEntryRX *entry)
|
||||
{
|
||||
return Bt(&entry->status, 0); // ?? TODO #define
|
||||
}
|
||||
|
||||
I64 E1000TransmitPacketAllocate(U8 **packet_buffer_out, I64 length)
|
||||
{ I64 de_index = e1000.current_tx_de_index;
|
||||
|
||||
CE1000DescriptorEntryTX *entry = &e1000.tx_de_buffer[de_index * sizeof(CE1000DescriptorEntryTX)];
|
||||
|
||||
*packet_buffer_out = e1000.tx_buffer_addr + de_index * ETHERNET_FRAME_SIZE;
|
||||
|
||||
MemSet(*packet_buffer_out, 0, ETHERNET_FRAME_SIZE); // Clear buffer contents in advance.
|
||||
|
||||
entry->address = *packet_buffer_out;
|
||||
entry->length = length;
|
||||
entry->cmd = (1 << 3) | 3; // ??? TODO #define
|
||||
|
||||
NetLog("E1000 ALLOCATE TX PACKET: de_index: %X.", de_index);
|
||||
return de_index;
|
||||
}
|
||||
|
||||
U0 E1000TransmitPacketFinish(I64 de_index)
|
||||
{
|
||||
CE1000DescriptorEntryTX *entry = &e1000.tx_de_buffer[de_index * sizeof(CE1000DescriptorEntryTX)];
|
||||
|
||||
e1000.current_tx_de_index = (e1000.current_tx_de_index + 1) & (E1000_TX_BUFF_COUNT - 1);
|
||||
E1000MMIOWrite(E1000_REG_TDT, e1000.current_tx_de_index);
|
||||
|
||||
ClassRep(entry);
|
||||
|
||||
while (!(entry->sta & 0xF)) // ???
|
||||
Yield;
|
||||
|
||||
NetLog("E1000 FINISH TX PACKET: TX DE index: %X.", de_index);
|
||||
}
|
||||
|
||||
U0 EthernetFrameFinish(I64 de_index)
|
||||
{//Alias for driver Finish TX function.
|
||||
E1000TransmitPacketFinish(de_index);
|
||||
}
|
||||
|
||||
interrupt U0 E1000IRQ()
|
||||
{ // TODO need #defines
|
||||
U32 icr = E1000MMIORead(0xC0); // ???
|
||||
Bool poll = FALSE;
|
||||
CE1000DescriptorEntryRX *entry = e1000.rx_de_buffer;
|
||||
U8 *packet_buffer;
|
||||
U16 packet_length;
|
||||
I64 de_index;
|
||||
|
||||
icr &= ~3; // ??
|
||||
|
||||
if (icr & (1 << 2)) // ?? 'link status change' ??
|
||||
{
|
||||
icr &= ~(1 << 2);
|
||||
E1000MMIOWrite(E1000_REG_CTRL, E1000MMIORead(E1000_REG_CTRL) | E1000_CTRLF_SLU);
|
||||
}
|
||||
|
||||
if (icr & (1 << 6) || icr & (1 << 4)) // ?? 'rx underrun / min threshold' ??
|
||||
{
|
||||
icr = ~((1 << 6) | (1 << 4)); // ??
|
||||
|
||||
poll = TRUE;
|
||||
}
|
||||
|
||||
if (icr & (1 << 7)) // ?? 'packet pending?
|
||||
{
|
||||
icr &= ~(1 << 7);
|
||||
|
||||
poll = TRUE;
|
||||
}
|
||||
|
||||
if (poll)
|
||||
{
|
||||
while (E1000DriverOwnsRX(&entry[e1000.current_rx_de_index]))
|
||||
{
|
||||
NetLog("$$BG,LTCYAN$$$$FG,WHITE$$"
|
||||
"==== E1000 IRQ ===="
|
||||
"$$BG$$$$FG$$");
|
||||
|
||||
NetLog("$$BD,CYAN$$$$FD,WHITE$$"
|
||||
"E1000 IRQ: Saw owned RX DE index %d.", e1000.current_rx_de_index);
|
||||
|
||||
de_index = E1000PacketReceive(&packet_buffer, &packet_length);
|
||||
|
||||
if (de_index >= 0)
|
||||
{
|
||||
NetLog("E1000 IRQ: Pushing copy into Net Queue, releasing receive packet.");
|
||||
NetQueuePush(packet_buffer, packet_length);
|
||||
E1000ReceivePacketRelease(de_index);
|
||||
}
|
||||
|
||||
NetLog("E1000 IRQ: Exiting.\n"
|
||||
"$$BD,WHITE$$$$FD,LTGRAY$$"
|
||||
"$$BG,LTCYAN$$$$FG,WHITE$$"
|
||||
"==================="
|
||||
"$$BG$$$$FG$$");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
E1000MMIORead(0xC0); // ?? 'clear pending interrupts' ??
|
||||
|
||||
*(dev.uncached_alias + LAPIC_EOI)(U32*) = 0;
|
||||
|
||||
}
|
||||
|
||||
/*
|
||||
U0 PCIInterruptsReroute(I64 base)
|
||||
{ // todo: comments explaining process, maybe better var names
|
||||
I64 i;
|
||||
|
@ -212,13 +367,15 @@ U0 PCIInterruptsReroute(I64 base)
|
|||
|
||||
|
||||
U0 E1000InterruptsSetup()
|
||||
{
|
||||
// PCIInterruptsReroute(I_E1000);
|
||||
NetErr("TODO E1000");
|
||||
{ // .. ?
|
||||
IntEntrySet(I_E1000_0, &E1000IRQ);
|
||||
IntEntrySet(I_E1000_1, &E1000IRQ);
|
||||
IntEntrySet(I_E1000_2, &E1000IRQ);
|
||||
IntEntrySet(I_E1000_3, &E1000IRQ);
|
||||
PCIInterruptsReroute(I_E1000_0);
|
||||
}
|
||||
*/
|
||||
|
||||
U0 E1000RXInit()
|
||||
U0 E1000InitRX()
|
||||
{
|
||||
I64 de_index;
|
||||
|
||||
|
@ -265,7 +422,7 @@ U0 E1000RXInit()
|
|||
E1000_RCTLF_BAM);
|
||||
}
|
||||
|
||||
U0 E1000TXInit()
|
||||
U0 E1000InitTX()
|
||||
{
|
||||
e1000.tx_de_buffer_phys = CAllocAligned(sizeof(CE1000DescriptorEntryTX) * E1000_TX_BUFF_COUNT,
|
||||
16,
|
||||
|
@ -333,8 +490,10 @@ U0 E1000Init()
|
|||
E1000MMIORead(0xC0); // TODO : WHAT IS THIS ???
|
||||
|
||||
// start rx tx?
|
||||
E1000RXInit;
|
||||
E1000TXInit;
|
||||
E1000InitRX;
|
||||
E1000InitTX;
|
||||
|
||||
E1000InterruptsSetup;
|
||||
|
||||
|
||||
NetErr("TODO E1000");
|
||||
|
@ -348,9 +507,34 @@ I64 EthernetFrameAllocate(U8 **packet_buffer_out,
|
|||
U16 ethertype,
|
||||
I64 packet_length)
|
||||
{
|
||||
no_warn packet_buffer_out, source_address, destination_address, ethertype, packet_length;
|
||||
NetErr("TODO E1000");
|
||||
return -1;
|
||||
U8 *ethernet_frame;
|
||||
I64 de_index;
|
||||
|
||||
//need to see if 3 years later VirtualBox supports APAD_XMT!
|
||||
if (packet_length < ETHERNET_MIN_FRAME_SIZE)
|
||||
{
|
||||
NetWarn("ETHERNET FRAME ALLOCATE: Truncating length");
|
||||
packet_length = ETHERNET_MIN_FRAME_SIZE;
|
||||
}
|
||||
|
||||
de_index = E1000TransmitPacketAllocate(ðernet_frame, ETHERNET_MAC_HEADER_LENGTH + packet_length);
|
||||
|
||||
if (de_index < 0)
|
||||
{
|
||||
NetErr("ETHERNET FRAME ALLOCATE: Failure");
|
||||
return -1; // Positive value expected. Functions calling this must factor this in.
|
||||
}
|
||||
|
||||
MemCopy(ethernet_frame, destination_address, MAC_ADDRESS_LENGTH);
|
||||
MemCopy(ethernet_frame + MAC_ADDRESS_LENGTH, source_address, MAC_ADDRESS_LENGTH);
|
||||
|
||||
ethernet_frame[ETHERNET_ETHERTYPE_OFFSET] = ethertype >> 8;
|
||||
ethernet_frame[ETHERNET_ETHERTYPE_OFFSET + 1] = ethertype & 0xFF;
|
||||
|
||||
*packet_buffer_out = ethernet_frame + ETHERNET_MAC_HEADER_LENGTH;
|
||||
|
||||
return de_index;
|
||||
|
||||
}
|
||||
|
||||
U8 *EthernetMACGet()
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
Cd(__DIR__);
|
||||
Cd(__DIR__);;
|
||||
#include "Utilities/Net.HH"
|
||||
|
||||
#include "Utilities/NetLog"
|
||||
|
|
|
@ -31,6 +31,12 @@
|
|||
#define I_PCNET1 I_USER + 1
|
||||
#define I_PCNET2 I_USER + 2
|
||||
#define I_PCNET3 I_USER + 3
|
||||
|
||||
#define I_E1000_0 I_USER + 0
|
||||
#define I_E1000_1 I_USER + 1
|
||||
#define I_E1000_2 I_USER + 2
|
||||
#define I_E1000_3 I_USER + 3
|
||||
|
||||
#define I_NETHANDLER I_USER + 4
|
||||
|
||||
#define INT_DEST_CPU 0
|
||||
|
|
Loading…
Reference in a new issue