2019-03-10 15:25:33 +01:00
|
|
|
#pragma once
|
|
|
|
|
2019-05-28 11:53:16 +02:00
|
|
|
#include <AK/OwnPtr.h>
|
|
|
|
#include <Kernel/IRQHandler.h>
|
2019-04-02 19:54:38 +02:00
|
|
|
#include <Kernel/Net/NetworkAdapter.h>
|
2019-03-10 15:25:33 +01:00
|
|
|
#include <Kernel/PCI.h>
|
2019-04-03 15:13:07 +02:00
|
|
|
#include <Kernel/VM/MemoryManager.h>
|
2019-03-10 15:25:33 +01:00
|
|
|
|
2019-05-28 11:53:16 +02:00
|
|
|
class E1000NetworkAdapter final : public NetworkAdapter
|
|
|
|
, public IRQHandler {
|
2019-03-10 15:25:33 +01:00
|
|
|
public:
|
2019-03-10 20:59:23 +01:00
|
|
|
static E1000NetworkAdapter* the();
|
|
|
|
|
2019-03-10 15:25:33 +01:00
|
|
|
static OwnPtr<E1000NetworkAdapter> autodetect();
|
|
|
|
|
|
|
|
E1000NetworkAdapter(PCI::Address, byte irq);
|
|
|
|
virtual ~E1000NetworkAdapter() override;
|
|
|
|
|
2019-03-10 23:40:09 +01:00
|
|
|
virtual void send_raw(const byte*, int) override;
|
2019-03-10 20:59:23 +01:00
|
|
|
|
2019-03-10 15:25:33 +01:00
|
|
|
private:
|
|
|
|
virtual void handle_irq() override;
|
|
|
|
virtual const char* class_name() const override { return "E1000NetworkAdapter"; }
|
|
|
|
|
2019-05-28 11:53:16 +02:00
|
|
|
struct [[gnu::packed]] e1000_rx_desc
|
|
|
|
{
|
2019-03-10 20:59:23 +01:00
|
|
|
volatile uint64_t addr { 0 };
|
|
|
|
volatile uint16_t length { 0 };
|
|
|
|
volatile uint16_t checksum { 0 };
|
|
|
|
volatile uint8_t status { 0 };
|
|
|
|
volatile uint8_t errors { 0 };
|
|
|
|
volatile uint16_t special { 0 };
|
|
|
|
};
|
|
|
|
|
2019-05-28 11:53:16 +02:00
|
|
|
struct [[gnu::packed]] e1000_tx_desc
|
|
|
|
{
|
2019-03-10 20:59:23 +01:00
|
|
|
volatile uint64_t addr { 0 };
|
|
|
|
volatile uint16_t length { 0 };
|
|
|
|
volatile uint8_t cso { 0 };
|
|
|
|
volatile uint8_t cmd { 0 };
|
|
|
|
volatile uint8_t status { 0 };
|
|
|
|
volatile uint8_t css { 0 };
|
|
|
|
volatile uint16_t special { 0 };
|
|
|
|
};
|
|
|
|
|
2019-03-10 15:25:33 +01:00
|
|
|
void detect_eeprom();
|
|
|
|
dword read_eeprom(byte address);
|
|
|
|
void read_mac_address();
|
|
|
|
|
|
|
|
void write_command(word address, dword);
|
|
|
|
dword read_command(word address);
|
|
|
|
|
2019-03-10 20:59:23 +01:00
|
|
|
void initialize_rx_descriptors();
|
|
|
|
void initialize_tx_descriptors();
|
|
|
|
|
2019-03-10 15:25:33 +01:00
|
|
|
void out8(word address, byte);
|
|
|
|
void out16(word address, word);
|
|
|
|
void out32(word address, dword);
|
|
|
|
byte in8(word address);
|
|
|
|
word in16(word address);
|
|
|
|
dword in32(word address);
|
|
|
|
|
2019-03-10 23:40:09 +01:00
|
|
|
void receive();
|
|
|
|
|
2019-03-10 15:25:33 +01:00
|
|
|
PCI::Address m_pci_address;
|
|
|
|
word m_io_base { 0 };
|
|
|
|
PhysicalAddress m_mmio_base;
|
|
|
|
byte m_interrupt_line { 0 };
|
|
|
|
bool m_has_eeprom { false };
|
|
|
|
bool m_use_mmio { false };
|
2019-03-10 20:59:23 +01:00
|
|
|
|
|
|
|
static const int number_of_rx_descriptors = 32;
|
|
|
|
static const int number_of_tx_descriptors = 8;
|
|
|
|
|
|
|
|
e1000_rx_desc* m_rx_descriptors;
|
|
|
|
e1000_tx_desc* m_tx_descriptors;
|
2019-03-10 15:25:33 +01:00
|
|
|
};
|