/* * Copyright 2008-2010, Axel Dörfler, axeld@pinc-software.de. * Copyright 2011-2016, Haiku, Inc. All Rights Reserved. * Distributed under the terms of the MIT License. * * Authors: * Axel Dörfler, axeld@pinc-software.de * Jerome Duval, jerome.duval@gmail.com * Adrien Destugues, pulkomandy@gmail.com * Michael Lotz, mmlr@mlotz.ch * Alexander von Gluck IV, kallisti5@unixzen.com */ #include #include #include #include #include #include #include #define TRACE_INTEL #ifdef TRACE_INTEL # define TRACE(x...) dprintf("intel_gart: " x) #else # define TRACE(x...) ; #endif #define ERROR(x...) dprintf("intel_gart: " x) /* read and write to PCI config space */ #define get_pci_config(info, offset, size) \ (sPCI->read_pci_config((info).bus, (info).device, (info).function, \ (offset), (size))) #define set_pci_config(info, offset, size, value) \ (sPCI->write_pci_config((info).bus, (info).device, (info).function, \ (offset), (size), (value))) #define write32(address, data) \ (*((volatile uint32*)(address)) = (data)) #define read32(address) \ (*((volatile uint32*)(address))) // PCI "Host bridge" is most cases :-) const struct supported_device { uint32 bridge_id; uint32 display_id; int32 type; const char *name; } kSupportedDevices[] = { {0x3575, 0x3577, INTEL_GROUP_83x, "i830GM"}, {0x2560, 0x2562, INTEL_GROUP_83x, "i845G"}, {0x3580, 0x3582, INTEL_GROUP_85x, "i855G"}, {0x358c, 0x358e, INTEL_GROUP_85x, "i855G"}, {0x2570, 0x2572, INTEL_GROUP_85x, "i865G"}, // {0x2792, INTEL_GROUP_91x, "i910"}, // {0x258a, INTEL_GROUP_91x, "i915"}, {0x2580, 0x2582, INTEL_MODEL_915, "i915G"}, {0x2590, 0x2592, INTEL_MODEL_915M, "i915GM"}, {0x2770, 0x2772, INTEL_MODEL_945, "i945G"}, {0x27a0, 0x27a2, INTEL_MODEL_945M, "i945GM"}, {0x27ac, 0x27ae, INTEL_MODEL_945M, "i945GME"}, {0x2970, 0x2972, INTEL_MODEL_965, "i946GZ"}, {0x2980, 0x2982, INTEL_MODEL_965, "G35"}, {0x2990, 0x2992, INTEL_MODEL_965, "i965Q"}, {0x29a0, 0x29a2, INTEL_MODEL_965, "i965G"}, {0x2a00, 0x2a02, INTEL_MODEL_965, "i965GM"}, {0x2a10, 0x2a12, INTEL_MODEL_965, "i965GME"}, {0x29b0, 0x29b2, INTEL_MODEL_G33, "G33"}, {0x29c0, 0x29c2, INTEL_MODEL_G33, "Q35"}, {0x29d0, 0x29d2, INTEL_MODEL_G33, "Q33"}, {0x2a40, 0x2a42, INTEL_MODEL_GM45, "GM45"}, {0x2e00, 0x2e02, INTEL_MODEL_G45, "IGD"}, {0x2e10, 0x2e12, INTEL_MODEL_G45, "Q45"}, {0x2e20, 0x2e22, INTEL_MODEL_G45, "G45"}, {0x2e30, 0x2e32, INTEL_MODEL_G45, "G41"}, {0x2e40, 0x2e42, INTEL_MODEL_G45, "B43"}, {0x2e90, 0x2e92, INTEL_MODEL_G45, "B43"}, {0xa000, 0xa001, INTEL_MODEL_PINE, "Atom D4xx"}, {0xa000, 0xa002, INTEL_MODEL_PINE, "Atom D5xx"}, {0xa010, 0xa011, INTEL_MODEL_PINEM, "Atom N4xx"}, {0xa010, 0xa012, INTEL_MODEL_PINEM, "Atom N5xx"}, {0x0040, 0x0042, INTEL_MODEL_ILKG, "IronLake Desktop"}, {0x0044, 0x0046, INTEL_MODEL_ILKGM, "IronLake Mobile"}, {0x0062, 0x0046, INTEL_MODEL_ILKGM, "IronLake Mobile"}, {0x006a, 0x0046, INTEL_MODEL_ILKGM, "IronLake Mobile"}, {0x0100, 0x0102, INTEL_MODEL_SNBG, "SandyBridge Desktop GT1"}, {0x0100, 0x0112, INTEL_MODEL_SNBG, "SandyBridge Desktop GT2"}, {0x0100, 0x0122, INTEL_MODEL_SNBG, "SandyBridge Desktop GT2+"}, {0x0104, 0x0106, INTEL_MODEL_SNBGM, "SandyBridge Mobile GT1"}, {0x0104, 0x0116, INTEL_MODEL_SNBGM, "SandyBridge Mobile GT2"}, {0x0104, 0x0126, INTEL_MODEL_SNBGM, "SandyBridge Mobile GT2+"}, {0x0108, 0x010a, INTEL_MODEL_SNBGS, "SandyBridge Server"}, {0x0150, 0x0152, INTEL_MODEL_IVBG, "IvyBridge Desktop GT1"}, {0x0150, 0x0162, INTEL_MODEL_IVBG, "IvyBridge Desktop GT2"}, {0x0154, 0x0156, INTEL_MODEL_IVBGM, "IvyBridge Mobile GT1"}, {0x0154, 0x0166, INTEL_MODEL_IVBGM, "IvyBridge Mobile GT2"}, {0x0158, 0x015a, INTEL_MODEL_IVBGS, "IvyBridge Server GT1"}, {0x0158, 0x016a, INTEL_MODEL_IVBGS, "IvyBridge Server GT2"}, {0x0c00, 0x0412, INTEL_MODEL_HAS, "Haswell Desktop"}, {0x0c04, 0x0416, INTEL_MODEL_HASM, "Haswell Mobile"}, {0x0d04, 0x0d26, INTEL_MODEL_HASM, "Haswell Mobile"}, {0x0a04, 0x0a16, INTEL_MODEL_HASM, "Haswell Mobile"}, #if 0 // XXX: 0x0f00 only confirmed on 0x0f30, 0x0f31 {0x0f00, 0x0f30, INTEL_MODEL_VLVM, "ValleyView Mobile"}, {0x0f00, 0x0f31, INTEL_MODEL_VLVM, "ValleyView Mobile"}, {0x0f00, 0x0f32, INTEL_MODEL_VLVM, "ValleyView Mobile"}, {0x0f00, 0x0f33, INTEL_MODEL_VLVM, "ValleyView Mobile"}, #endif // XXX: 0x1604 only confirmed on 0x1616 {0x1604, 0x1606, INTEL_MODEL_BDWM, "Broadwell GT1 ULT"}, {0x1604, 0x160b, INTEL_MODEL_BDWM, "Broadwell GT1 Iris"}, {0x1604, 0x160e, INTEL_MODEL_BDWM, "Broadwell GT1 ULX"}, {0x1604, 0x1602, INTEL_MODEL_BDWM, "Broadwell GT1 ULT"}, {0x1604, 0x160a, INTEL_MODEL_BDWS, "Broadwell GT1 Server"}, {0x1604, 0x160d, INTEL_MODEL_BDW, "Broadwell GT1 Workstation"}, {0x1604, 0x1616, INTEL_MODEL_BDWM, "Broadwell GT2 ULT"}, {0x1604, 0x161b, INTEL_MODEL_BDWM, "Broadwell GT2 ULT"}, {0x1604, 0x161e, INTEL_MODEL_BDWM, "Broadwell GT2 ULX"}, {0x1604, 0x1612, INTEL_MODEL_BDWM, "Broadwell GT2 Halo"}, {0x1604, 0x161a, INTEL_MODEL_BDWS, "Broadwell GT2 Server"}, {0x1604, 0x161d, INTEL_MODEL_BDW, "Broadwell GT2 Workstation"}, {0x1604, 0x1626, INTEL_MODEL_BDWM, "Broadwell GT3 ULT"}, {0x1604, 0x162b, INTEL_MODEL_BDWM, "Broadwell GT3 Iris"}, {0x1604, 0x162e, INTEL_MODEL_BDWM, "Broadwell GT3 ULX"}, {0x1604, 0x1622, INTEL_MODEL_BDWM, "Broadwell GT3 ULT"}, {0x1604, 0x162a, INTEL_MODEL_BDWS, "Broadwell GT3 Server"}, {0x1604, 0x162d, INTEL_MODEL_BDW, "Broadwell GT3 Workstation"}, // XXX: 0x1904 only confirmed on 0x1916 {0x1904, 0x1902, INTEL_MODEL_SKY, "Skylake GT1"}, {0x1904, 0x1906, INTEL_MODEL_SKYM, "Skylake GT1"}, {0x1904, 0x190a, INTEL_MODEL_SKYS, "Skylake GT1"}, {0x1904, 0x190b, INTEL_MODEL_SKY, "Skylake GT1"}, {0x1904, 0x190e, INTEL_MODEL_SKYM, "Skylake GT1"}, {0x191f, 0x1912, INTEL_MODEL_SKY, "Skylake GT2"}, // confirmed {0x1904, 0x1916, INTEL_MODEL_SKYM, "Skylake GT2"}, {0x1904, 0x191a, INTEL_MODEL_SKYS, "Skylake GT2"}, {0x1904, 0x191b, INTEL_MODEL_SKY, "Skylake GT2"}, {0x1904, 0x191d, INTEL_MODEL_SKY, "Skylake GT2"}, {0x1904, 0x191e, INTEL_MODEL_SKYM, "Skylake GT2"}, {0x1904, 0x1921, INTEL_MODEL_SKYM, "Skylake GT2F"}, {0x1904, 0x1926, INTEL_MODEL_SKYM, "Skylake GT3"}, {0x1904, 0x192a, INTEL_MODEL_SKYS, "Skylake GT3"}, {0x1904, 0x192b, INTEL_MODEL_SKY, "Skylake GT3"}, {0x5904, 0x5906, INTEL_MODEL_KBY, "Kabylake ULT GT1"}, {0x590f, 0x5902, INTEL_MODEL_KBY, "Kabylake DT GT1"}, {0x5904, 0x5916, INTEL_MODEL_KBYM, "Kabylake ULT GT2"}, {0x590c, 0x5916, INTEL_MODEL_KBYM, "Kabylake ULT GT2"}, {0x5904, 0x5921, INTEL_MODEL_KBYM, "Kabylake ULT GT2F"}, {0x590c, 0x591c, INTEL_MODEL_KBY, "Kabylake ULX GT2"}, {0x590c, 0x591e, INTEL_MODEL_KBY, "Kabylake ULX GT2"}, {0x591f, 0x5912, INTEL_MODEL_KBY, "Kabylake DT GT2"}, {0x5914, 0x5917, INTEL_MODEL_KBYM, "Kabylake Mobile GT2"}, {0x5910, 0x591b, INTEL_MODEL_KBYM, "Kabylake Halo GT2"}, {0x5918, 0x591d, INTEL_MODEL_KBY, "Kabylake WKS GT2"}, {0x5904, 0x5926, INTEL_MODEL_KBY, "Kabylake ULT GT3"}, {0x5904, 0x5927, INTEL_MODEL_KBY, "Kabylake ULT GT3"}, {0x3e0f, 0x3e90, INTEL_MODEL_CFL, "CoffeeLake GT1"}, {0x3e0f, 0x3e93, INTEL_MODEL_CFL, "CoffeeLake GT1"}, {0x3e1f, 0x3e91, INTEL_MODEL_CFL, "CoffeeLake GT2"}, {0x3ec2, 0x3e92, INTEL_MODEL_CFL, "CoffeeLake GT2"}, {0x3e18, 0x3e96, INTEL_MODEL_CFL, "CoffeeLake GT2"}, {0x3e30, 0x3e98, INTEL_MODEL_CFL, "CoffeeLake GT2"}, {0x3e31, 0x3e9a, INTEL_MODEL_CFL, "CoffeeLake GT2"}, {0x3ec4, 0x3e9b, INTEL_MODEL_CFLM, "CoffeeLake Halo GT2"}, {0x3e10, 0x3eab, INTEL_MODEL_CFLM, "CoffeeLake Halo GT2"}, {0x3ec4, 0x3eab, INTEL_MODEL_CFLM, "CoffeeLake Halo GT2"}, {0x3ed0, 0x3ea5, INTEL_MODEL_CFL, "CoffeeLake GT3"}, {0x3ed0, 0x3ea6, INTEL_MODEL_CFL, "CoffeeLake GT3"}, {0x9b64, 0x9ba4, INTEL_MODEL_CML, "CometLake GT1"}, {0x9b73, 0x9ba8, INTEL_MODEL_CML, "CometLake GT1"}, {0x9b71, 0x9b21, INTEL_MODEL_CMLM, "CometLake U GT1"}, {0x9b71, 0x9baa, INTEL_MODEL_CMLM, "CometLake U GT1"}, {0x9b54, 0x9bc4, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b43, 0x9bc5, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b53, 0x9bc5, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b33, 0x9bc6, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b53, 0x9bc8, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b63, 0x9bc8, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b53, 0x9be6, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b44, 0x9bf6, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b54, 0x9bf6, INTEL_MODEL_CML, "CometLake GT2"}, {0x9b61, 0x9b41, INTEL_MODEL_CMLM, "CometLake U GT2"}, {0x9b51, 0x9bca, INTEL_MODEL_CMLM, "CometLake U GT2"}, {0x9b61, 0x9bca, INTEL_MODEL_CMLM, "CometLake U GT2"}, {0x9b51, 0x9bcc, INTEL_MODEL_CMLM, "CometLake U GT2"}, {0x4e22, 0x4e55, INTEL_MODEL_JSL, "JasperLake"}, {0x4e24, 0x4e55, INTEL_MODEL_JSL, "JasperLake"}, {0x4e12, 0x4e61, INTEL_MODEL_JSL, "JasperLake"}, {0x4e26, 0x4e71, INTEL_MODEL_JSLM, "JasperLake"}, {0x4e28, 0x4e71, INTEL_MODEL_JSLM, "JasperLake"}, {0x9a14, 0x9a49, INTEL_MODEL_TGLM, "TigerLake-LP GT2"}, }; struct intel_info { pci_info bridge; pci_info display; DeviceType* type; uint32* gtt_base; phys_addr_t gtt_physical_base; area_id gtt_area; size_t gtt_entries; size_t gtt_stolen_entries; vuint32* registers; area_id registers_area; addr_t aperture_base; phys_addr_t aperture_physical_base; area_id aperture_area; size_t aperture_size; size_t aperture_stolen_size; phys_addr_t scratch_page; area_id scratch_area; }; static intel_info sInfo; static pci_module_info* sPCI; static bool has_display_device(pci_info &info, uint32 deviceID) { for (uint32 index = 0; sPCI->get_nth_pci_info(index, &info) == B_OK; index++) { if (info.vendor_id != VENDOR_ID_INTEL || info.device_id != deviceID || info.class_base != PCI_display) continue; return true; } return false; } static uint16 gtt_memory_config(intel_info &info) { uint8 controlRegister = INTEL_GRAPHICS_MEMORY_CONTROL; if (info.type->Generation() >= 6) controlRegister = SNB_GRAPHICS_MEMORY_CONTROL; return get_pci_config(info.bridge, controlRegister, 2); } static size_t determine_gtt_stolen(intel_info &info) { uint16 memoryConfig = gtt_memory_config(info); size_t memorySize = 1 << 20; // 1 MB if (info.type->InGroup(INTEL_GROUP_83x)) { // Older chips switch (memoryConfig & STOLEN_MEMORY_MASK) { case i830_LOCAL_MEMORY_ONLY: // TODO: determine its size! ERROR("getting local memory size not implemented.\n"); break; case i830_STOLEN_512K: memorySize >>= 1; break; case i830_STOLEN_1M: // default case break; case i830_STOLEN_8M: memorySize *= 8; break; } } else if (info.type->InGroup(INTEL_GROUP_SNB) || info.type->InGroup(INTEL_GROUP_IVB) || info.type->InGroup(INTEL_GROUP_HAS)) { switch (memoryConfig & SNB_STOLEN_MEMORY_MASK) { case SNB_STOLEN_MEMORY_32MB: memorySize *= 32; break; case SNB_STOLEN_MEMORY_64MB: memorySize *= 64; break; case SNB_STOLEN_MEMORY_96MB: memorySize *= 96; break; case SNB_STOLEN_MEMORY_128MB: memorySize *= 128; break; case SNB_STOLEN_MEMORY_160MB: memorySize *= 160; break; case SNB_STOLEN_MEMORY_192MB: memorySize *= 192; break; case SNB_STOLEN_MEMORY_224MB: memorySize *= 224; break; case SNB_STOLEN_MEMORY_256MB: memorySize *= 256; break; case SNB_STOLEN_MEMORY_288MB: memorySize *= 288; break; case SNB_STOLEN_MEMORY_320MB: memorySize *= 320; break; case SNB_STOLEN_MEMORY_352MB: memorySize *= 352; break; case SNB_STOLEN_MEMORY_384MB: memorySize *= 384; break; case SNB_STOLEN_MEMORY_416MB: memorySize *= 416; break; case SNB_STOLEN_MEMORY_448MB: memorySize *= 448; break; case SNB_STOLEN_MEMORY_480MB: memorySize *= 480; break; case SNB_STOLEN_MEMORY_512MB: memorySize *= 512; break; } } else if (info.type->InGroup(INTEL_GROUP_BDW) || info.type->InFamily(INTEL_FAMILY_LAKE)) { switch (memoryConfig & BDW_STOLEN_MEMORY_MASK) { case BDW_STOLEN_MEMORY_32MB: memorySize *= 32; break; case BDW_STOLEN_MEMORY_64MB: memorySize *= 64; break; case BDW_STOLEN_MEMORY_96MB: memorySize *= 96; break; case BDW_STOLEN_MEMORY_128MB: memorySize *= 128; break; case BDW_STOLEN_MEMORY_160MB: memorySize *= 160; break; case BDW_STOLEN_MEMORY_192MB: memorySize *= 192; break; case BDW_STOLEN_MEMORY_224MB: memorySize *= 224; break; case BDW_STOLEN_MEMORY_256MB: memorySize *= 256; break; case BDW_STOLEN_MEMORY_288MB: memorySize *= 288; break; case BDW_STOLEN_MEMORY_320MB: memorySize *= 320; break; case BDW_STOLEN_MEMORY_352MB: memorySize *= 352; break; case BDW_STOLEN_MEMORY_384MB: memorySize *= 384; break; case BDW_STOLEN_MEMORY_416MB: memorySize *= 416; break; case BDW_STOLEN_MEMORY_448MB: memorySize *= 448; break; case BDW_STOLEN_MEMORY_480MB: memorySize *= 480; break; case BDW_STOLEN_MEMORY_512MB: memorySize *= 512; break; case BDW_STOLEN_MEMORY_1024MB: memorySize *= 1024; break; case BDW_STOLEN_MEMORY_1536MB: memorySize *= 1536; break; } if(info.type->InGroup(INTEL_GROUP_BDW)) { if((memoryConfig & BDW_STOLEN_MEMORY_MASK) == BDW_STOLEN_MEMORY_2016MB) { memorySize *= 2016; } } else if(info.type->InFamily(INTEL_FAMILY_LAKE)) { switch(memoryConfig & BDW_STOLEN_MEMORY_MASK) { case SKL_STOLEN_MEMORY_4MB: memorySize *= 4; break; case SKL_STOLEN_MEMORY_8MB: memorySize *= 8; break; case SKL_STOLEN_MEMORY_12MB: memorySize *= 12; break; case SKL_STOLEN_MEMORY_16MB: memorySize *= 16; break; case SKL_STOLEN_MEMORY_20MB: memorySize *= 20; break; case SKL_STOLEN_MEMORY_24MB: memorySize *= 24; break; case SKL_STOLEN_MEMORY_28MB: memorySize *= 28; break; case SKL_STOLEN_MEMORY_32MB: memorySize *= 32; break; case SKL_STOLEN_MEMORY_36MB: memorySize *= 36; break; case SKL_STOLEN_MEMORY_40MB: memorySize *= 40; break; case SKL_STOLEN_MEMORY_44MB: memorySize *= 44; break; case SKL_STOLEN_MEMORY_48MB: memorySize *= 48; break; case SKL_STOLEN_MEMORY_52MB: memorySize *= 52; break; case SKL_STOLEN_MEMORY_56MB: memorySize *= 56; break; case SKL_STOLEN_MEMORY_60MB: memorySize *= 60; break; } } } else if (info.type->InGroup(INTEL_GROUP_85x) || info.type->InFamily(INTEL_FAMILY_9xx) || info.type->InGroup(INTEL_GROUP_ILK)) { switch (memoryConfig & STOLEN_MEMORY_MASK) { case i855_STOLEN_MEMORY_4M: memorySize *= 4; break; case i855_STOLEN_MEMORY_8M: memorySize *= 8; break; case i855_STOLEN_MEMORY_16M: memorySize *= 16; break; case i855_STOLEN_MEMORY_32M: memorySize *= 32; break; case i855_STOLEN_MEMORY_48M: memorySize *= 48; break; case i855_STOLEN_MEMORY_64M: memorySize *= 64; break; case i855_STOLEN_MEMORY_128M: memorySize *= 128; break; case i855_STOLEN_MEMORY_256M: memorySize *= 256; break; case G4X_STOLEN_MEMORY_96MB: memorySize *= 96; break; case G4X_STOLEN_MEMORY_160MB: memorySize *= 160; break; case G4X_STOLEN_MEMORY_224MB: memorySize *= 224; break; case G4X_STOLEN_MEMORY_352MB: memorySize *= 352; break; } } else { // TODO: error out! memorySize = 4096; } return memorySize - 4096; } static size_t determine_gtt_size(intel_info &info) { uint16 memoryConfig = gtt_memory_config(info); size_t gttSize = 0; if (info.type->IsModel(INTEL_MODEL_965)) { switch (memoryConfig & i965_GTT_MASK) { case i965_GTT_128K: gttSize = 128 << 10; break; case i965_GTT_256K: gttSize = 256 << 10; break; case i965_GTT_512K: gttSize = 512 << 10; break; } } else if (info.type->IsModel(INTEL_MODEL_G33) || info.type->InGroup(INTEL_GROUP_PIN)) { switch (memoryConfig & G33_GTT_MASK) { case G33_GTT_1M: gttSize = 1 << 20; break; case G33_GTT_2M: gttSize = 2 << 20; break; } } else if (info.type->InGroup(INTEL_GROUP_G4x) || info.type->InGroup(INTEL_GROUP_ILK)) { switch (memoryConfig & G4X_GTT_MASK) { case G4X_GTT_NONE: gttSize = 0; break; case G4X_GTT_1M_NO_IVT: gttSize = 1 << 20; break; case G4X_GTT_2M_NO_IVT: case G4X_GTT_2M_IVT: gttSize = 2 << 20; break; case G4X_GTT_3M_IVT: gttSize = 3 << 20; break; case G4X_GTT_4M_IVT: gttSize = 4 << 20; break; } } else if (info.type->InGroup(INTEL_GROUP_SNB) || info.type->InGroup(INTEL_GROUP_IVB) || info.type->InGroup(INTEL_GROUP_HAS)) { switch (memoryConfig & SNB_GTT_SIZE_MASK) { case SNB_GTT_SIZE_NONE: gttSize = 0; break; case SNB_GTT_SIZE_1MB: gttSize = 1 << 20; break; case SNB_GTT_SIZE_2MB: gttSize = 2 << 20; break; } } else if (info.type->InGroup(INTEL_GROUP_BDW) || info.type->InFamily(INTEL_FAMILY_LAKE)) { switch (memoryConfig & BDW_GTT_SIZE_MASK) { case BDW_GTT_SIZE_NONE: gttSize = 0; break; case BDW_GTT_SIZE_2MB: gttSize = 2 << 20; break; case BDW_GTT_SIZE_4MB: gttSize = 4 << 20; break; case BDW_GTT_SIZE_8MB: gttSize = 8 << 20; break; } } else { // older models have the GTT as large as their frame buffer mapping // TODO: check if the i9xx version works with the i8xx chips as well size_t frameBufferSize = 0; if (info.type->InFamily(INTEL_FAMILY_8xx)) { if (info.type->InGroup(INTEL_GROUP_83x) && (memoryConfig & MEMORY_MASK) == i830_FRAME_BUFFER_64M) frameBufferSize = 64 << 20; else frameBufferSize = 128 << 20; } else if (info.type->Generation() >= 3) { frameBufferSize = info.display.u.h0.base_register_sizes[2]; } TRACE("frame buffer size %lu MB\n", frameBufferSize >> 20); gttSize = frameBufferSize / 1024; } return gttSize; } static void set_gtt_entry(intel_info &info, uint32 offset, phys_addr_t physicalAddress) { if (info.type->Generation() >= 8) { // CHV + BXT physicalAddress |= (physicalAddress >> 28) & 0x07f0; // TODO: cache control? } else if (info.type->Generation() >= 6) { // SandyBridge, IronLake, IvyBridge, Haswell physicalAddress |= (physicalAddress >> 28) & 0x0ff0; physicalAddress |= 0x02; // cache control, l3 cacheable } else if (info.type->Generation() >= 4) { // Intel 9xx minus 91x, 94x, G33 // possible high bits are stored in the lower end physicalAddress |= (physicalAddress >> 28) & 0x00f0; // TODO: cache control? } // TODO: this is not 64-bit safe! write32(info.gtt_base + (offset >> GTT_PAGE_SHIFT), (uint32)physicalAddress | GTT_ENTRY_VALID); } static void intel_unmap(intel_info &info) { delete_area(info.registers_area); delete_area(info.gtt_area); delete_area(info.scratch_area); delete_area(info.aperture_area); info.aperture_size = 0; } static status_t intel_map(intel_info &info) { int fbIndex = 0; int mmioIndex = 1; if (info.type->Generation() >= 3) { // for some reason Intel saw the need to change the order of the // mappings with the introduction of the i9xx family mmioIndex = 0; fbIndex = 2; } phys_addr_t addr = info.display.u.h0.base_registers[mmioIndex]; uint64 barSize = info.display.u.h0.base_register_sizes[mmioIndex]; if ((info.display.u.h0.base_register_flags[mmioIndex] & PCI_address_type) == PCI_address_type_64) { addr |= (uint64)info.display.u.h0.base_registers[mmioIndex + 1] << 32; barSize |= (uint64)info.display.u.h0.base_register_sizes[mmioIndex + 1] << 32; } AreaKeeper mmioMapper; info.registers_area = mmioMapper.Map("intel GMCH mmio", addr, barSize, B_ANY_KERNEL_ADDRESS, B_KERNEL_READ_AREA | B_KERNEL_WRITE_AREA, (void**)&info.registers); if (mmioMapper.InitCheck() < B_OK) { ERROR("could not map memory I/O!\n"); return info.registers_area; } // make sure bus master, memory-mapped I/O, and frame buffer is enabled set_pci_config(info.display, PCI_command, 2, get_pci_config(info.display, PCI_command, 2) | PCI_command_io | PCI_command_memory | PCI_command_master); void* scratchAddress; AreaKeeper scratchCreator; info.scratch_area = scratchCreator.Create("intel GMCH scratch", &scratchAddress, B_ANY_KERNEL_ADDRESS, B_PAGE_SIZE, B_FULL_LOCK, B_KERNEL_READ_AREA | B_KERNEL_WRITE_AREA); if (scratchCreator.InitCheck() < B_OK) { ERROR("could not create scratch page!\n"); return info.scratch_area; } physical_entry entry; if (get_memory_map(scratchAddress, B_PAGE_SIZE, &entry, 1) != B_OK) return B_ERROR; // TODO: Review these if (info.type->InFamily(INTEL_FAMILY_8xx)) { info.gtt_physical_base = read32(info.registers + INTEL_PAGE_TABLE_CONTROL) & ~PAGE_TABLE_ENABLED; if (info.gtt_physical_base == 0) { // TODO: not sure how this is supposed to work under Linux/FreeBSD, // but on my i865, this code is needed for Haiku. ERROR("Use GTT address fallback.\n"); info.gtt_physical_base = info.display.u.h0.base_registers[mmioIndex] + i830_GTT_BASE; } } else if (info.type->InGroup(INTEL_GROUP_91x)) { info.gtt_physical_base = get_pci_config(info.display, i915_GTT_BASE, 4); } else { // 945+? info.gtt_physical_base = addr + (2UL << 20); } size_t gttSize = determine_gtt_size(info); size_t stolenSize = determine_gtt_stolen(info); info.gtt_entries = gttSize / 4096; info.gtt_stolen_entries = stolenSize / 4096; TRACE("GTT base %" B_PRIxPHYSADDR ", size %lu, entries %lu, stolen %lu\n", info.gtt_physical_base, gttSize, info.gtt_entries, stolenSize); AreaKeeper gttMapper; info.gtt_area = gttMapper.Map("intel GMCH gtt", info.gtt_physical_base, gttSize, B_ANY_KERNEL_ADDRESS, B_KERNEL_READ_AREA | B_KERNEL_WRITE_AREA, (void**)&info.gtt_base); if (gttMapper.InitCheck() < B_OK) { ERROR("could not map GTT!\n"); return info.gtt_area; } info.aperture_physical_base = info.display.u.h0.base_registers[fbIndex]; info.aperture_stolen_size = stolenSize; if ((info.display.u.h0.base_register_flags[fbIndex] & PCI_address_type) == PCI_address_type_64) { info.aperture_physical_base |= (uint64)info.display.u.h0.base_registers[fbIndex + 1] << 32; if (info.aperture_size == 0) { info.aperture_size = info.display.u.h0.base_register_sizes[fbIndex] |= (uint64)info.display.u.h0.base_register_sizes[fbIndex + 1] << 32; } } else if (info.aperture_size == 0) info.aperture_size = info.display.u.h0.base_register_sizes[fbIndex]; ERROR("detected %ld MB of stolen memory, aperture size %ld MB, " "GTT size %ld KB\n", (stolenSize + (1023 << 10)) >> 20, info.aperture_size >> 20, gttSize >> 10); ERROR("GTT base = 0x%" B_PRIxPHYSADDR "\n", info.gtt_physical_base); ERROR("MMIO base = 0x%" B_PRIx32 "\n", info.display.u.h0.base_registers[mmioIndex]); ERROR("GMR base = 0x%" B_PRIxPHYSADDR "\n", info.aperture_physical_base); AreaKeeper apertureMapper; info.aperture_area = apertureMapper.Map("intel graphics aperture", info.aperture_physical_base, info.aperture_size, B_ANY_KERNEL_BLOCK_ADDRESS | B_MTR_WC, B_READ_AREA | B_WRITE_AREA, (void**)&info.aperture_base); if (apertureMapper.InitCheck() < B_OK) { // try again without write combining ERROR("enabling write combined mode failed.\n"); info.aperture_area = apertureMapper.Map("intel graphics aperture", info.aperture_physical_base, info.aperture_size, B_ANY_KERNEL_BLOCK_ADDRESS, B_READ_AREA | B_WRITE_AREA, (void**)&info.aperture_base); } if (apertureMapper.InitCheck() < B_OK) { ERROR("could not map graphics aperture!\n"); return info.aperture_area; } info.scratch_page = entry.address; gttMapper.Detach(); mmioMapper.Detach(); scratchCreator.Detach(); apertureMapper.Detach(); return B_OK; } // #pragma mark - module interface status_t intel_create_aperture(uint8 bus, uint8 device, uint8 function, size_t size, void** _aperture) { // TODO: we currently only support a single AGP bridge! if ((bus != sInfo.bridge.bus || device != sInfo.bridge.device || function != sInfo.bridge.function) && (bus != sInfo.display.bus || device != sInfo.display.device || function != sInfo.display.function)) return B_BAD_VALUE; sInfo.aperture_size = size; if (intel_map(sInfo) < B_OK) return B_ERROR; uint16 gmchControl = get_pci_config(sInfo.bridge, INTEL_GRAPHICS_MEMORY_CONTROL, 2) | MEMORY_CONTROL_ENABLED; set_pci_config(sInfo.bridge, INTEL_GRAPHICS_MEMORY_CONTROL, 2, gmchControl); write32(sInfo.registers + INTEL_PAGE_TABLE_CONTROL, sInfo.gtt_physical_base | PAGE_TABLE_ENABLED); read32(sInfo.registers + INTEL_PAGE_TABLE_CONTROL); if (sInfo.scratch_page != 0) { for (size_t i = sInfo.gtt_stolen_entries; i < sInfo.gtt_entries; i++) { set_gtt_entry(sInfo, i << GTT_PAGE_SHIFT, sInfo.scratch_page); } read32(sInfo.gtt_base + sInfo.gtt_entries - 1); } asm("wbinvd;"); *_aperture = NULL; return B_OK; } void intel_delete_aperture(void* aperture) { intel_unmap(sInfo); } static status_t intel_get_aperture_info(void* aperture, aperture_info* info) { if (info == NULL) return B_BAD_VALUE; info->base = sInfo.aperture_base; info->physical_base = sInfo.aperture_physical_base; info->size = sInfo.aperture_size; info->reserved_size = sInfo.aperture_stolen_size; return B_OK; } status_t intel_set_aperture_size(void* aperture, size_t size) { return B_ERROR; } static status_t intel_bind_page(void* aperture, uint32 offset, phys_addr_t physicalAddress) { //TRACE("bind_page(offset %lx, physical %lx)\n", offset, physicalAddress); set_gtt_entry(sInfo, offset, physicalAddress); return B_OK; } static status_t intel_unbind_page(void* aperture, uint32 offset) { //TRACE("unbind_page(offset %lx)\n", offset); if (sInfo.scratch_page != 0) set_gtt_entry(sInfo, offset, sInfo.scratch_page); return B_OK; } void intel_flush_tlbs(void* aperture) { read32(sInfo.gtt_base + sInfo.gtt_entries - 1); asm("wbinvd;"); } // #pragma mark - static status_t intel_init() { TRACE("bus manager init\n"); if (get_module(B_PCI_MODULE_NAME, (module_info**)&sPCI) != B_OK) return B_ERROR; for (uint32 index = 0; sPCI->get_nth_pci_info(index, &sInfo.bridge) == B_OK; index++) { if (sInfo.bridge.vendor_id != VENDOR_ID_INTEL || sInfo.bridge.class_base != PCI_bridge) continue; // check device for (uint32 i = 0; i < sizeof(kSupportedDevices) / sizeof(kSupportedDevices[0]); i++) { if (sInfo.bridge.device_id == kSupportedDevices[i].bridge_id) { sInfo.type = new DeviceType(kSupportedDevices[i].type); if (has_display_device(sInfo.display, kSupportedDevices[i].display_id)) { TRACE("found intel bridge\n"); return B_OK; } } } } return ENODEV; } static void intel_uninit() { if (sInfo.type) delete sInfo.type; } static int32 intel_std_ops(int32 op, ...) { switch (op) { case B_MODULE_INIT: return intel_init(); case B_MODULE_UNINIT: intel_uninit(); return B_OK; } return B_BAD_VALUE; } static struct agp_gart_bus_module_info sIntelModuleInfo = { { "busses/agp_gart/intel/v0", 0, intel_std_ops }, intel_create_aperture, intel_delete_aperture, intel_get_aperture_info, intel_set_aperture_size, intel_bind_page, intel_unbind_page, intel_flush_tlbs }; module_info* modules[] = { (module_info*)&sIntelModuleInfo, NULL };