mirror of
https://github.com/CalcProgrammer1/OpenRGB.git
synced 2025-12-25 16:27:50 -05:00
230 lines
9.2 KiB
C++
230 lines
9.2 KiB
C++
#include "Detector.h"
|
|
#include "CrucialController.h"
|
|
#include "LogManager.h"
|
|
#include "RGBController.h"
|
|
#include "RGBController_Crucial.h"
|
|
#include "i2c_smbus.h"
|
|
#include "pci_ids.h"
|
|
#include <vector>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
using namespace std::chrono_literals;
|
|
|
|
/*----------------------------------------------------------------------*\
|
|
| This list contains the available SMBus addresses for Crucial RAM |
|
|
\*----------------------------------------------------------------------*/
|
|
#define CRUCIAL_ADDRESS_COUNT 8
|
|
|
|
static const unsigned char crucial_addresses[] =
|
|
{
|
|
/*-----------------------------------------------------*\
|
|
| These addresses have been disabled due to conflict |
|
|
| with ASUS Aura DRAM. Since the detection scheme is |
|
|
| the same, Aura RAM will be detected as Crucial. |
|
|
| We need to improve the Crucial detection scheme. |
|
|
\*-----------------------------------------------------*/
|
|
0x39,
|
|
0x3A,
|
|
0x3B,
|
|
0x3C,
|
|
0x20,
|
|
0x21,
|
|
0x22,
|
|
0x23
|
|
};
|
|
|
|
#define CRUCIAL_CONTROLLER_NAME "Crucial DRAM"
|
|
std::string concatHexArray(const unsigned char array[], int count, const char split_char[])
|
|
{
|
|
std::string addresses = "";
|
|
for(int i = 0; i < count; i++)
|
|
{
|
|
char buffer[6];
|
|
snprintf(buffer, 6, "0x%02X%s", array[i], (i < count-1)? split_char: "");
|
|
addresses += buffer;
|
|
}
|
|
return addresses;
|
|
}
|
|
#define TESTING_ADDRESSES concatHexArray(crucial_addresses, CRUCIAL_ADDRESS_COUNT, "|").c_str()
|
|
|
|
/******************************************************************************************\
|
|
* *
|
|
* CrucialRegisterRead *
|
|
* *
|
|
* A standalone version of the AuraSMBusController::AuraRegisterRead function for *
|
|
* access to Aura devices without instancing the AuraSMBusController class or reading *
|
|
* the config table from the device. *
|
|
* *
|
|
\******************************************************************************************/
|
|
|
|
unsigned char CrucialRegisterRead(i2c_smbus_interface* bus, crucial_dev_id dev, crucial_register reg)
|
|
{
|
|
//Write Aura register
|
|
bus->i2c_smbus_write_word_data(dev, 0x00, ((reg << 8) & 0xFF00) | ((reg >> 8) & 0x00FF));
|
|
|
|
//Read Aura value
|
|
return(bus->i2c_smbus_read_byte_data(dev, 0x81));
|
|
}
|
|
|
|
/******************************************************************************************\
|
|
* *
|
|
* TestForCrucialController *
|
|
* *
|
|
* Tests the given address to see if an Crucial controller exists there. First does a*
|
|
* quick write to test for a response, and if so does a simple read at 0xA0 to test *
|
|
* for incrementing values 0...F which was observed at this location during data dump *
|
|
* *
|
|
\******************************************************************************************/
|
|
|
|
bool TestForCrucialController(i2c_smbus_interface* bus, unsigned char address)
|
|
{
|
|
bool pass = false;
|
|
|
|
int res = bus->i2c_smbus_write_quick(address, I2C_SMBUS_WRITE);
|
|
|
|
if (res >= 0)
|
|
{
|
|
pass = true;
|
|
|
|
LOG_DEBUG("[%s] Detected an I2C device at address %02X", CRUCIAL_CONTROLLER_NAME, address);
|
|
|
|
for (int i = 0xA0; i < 0xB0; i++)
|
|
{
|
|
res = bus->i2c_smbus_read_byte_data(address, i);
|
|
|
|
if (res != (i - 0xA0))
|
|
{
|
|
LOG_VERBOSE("[%s] Detection failed testing register %02X. Expected %02X, got %02X.", CRUCIAL_CONTROLLER_NAME, i, (i - 0xA0), res);
|
|
|
|
pass = false;
|
|
}
|
|
}
|
|
|
|
if(pass)
|
|
{
|
|
LOG_DEBUG("[%s] Checking for Micron string", CRUCIAL_CONTROLLER_NAME);
|
|
|
|
char buf[16];
|
|
for(int i = 0; i < 16; i++)
|
|
{
|
|
buf[i] = CrucialRegisterRead(bus, address, CRUCIAL_REG_MICRON_CHECK_1 + i);
|
|
}
|
|
|
|
if(strcmp(buf, "Micron") == 0)
|
|
{
|
|
LOG_DEBUG("[%s] Device %02X is a Micron device, continuing", CRUCIAL_CONTROLLER_NAME, address);
|
|
}
|
|
else
|
|
{
|
|
for(int i = 0; i < 16; i++)
|
|
{
|
|
buf[i] = CrucialRegisterRead(bus, address, CRUCIAL_REG_MICRON_CHECK_2 + i);
|
|
}
|
|
|
|
if(strcmp(buf, "Micron") == 0)
|
|
{
|
|
LOG_DEBUG("[%s] Device %02X is a Micron device, continuing", CRUCIAL_CONTROLLER_NAME, address);
|
|
}
|
|
else
|
|
{
|
|
LOG_DEBUG("[%s] Device %02X is not a Micron device, skipping", CRUCIAL_CONTROLLER_NAME, address);
|
|
pass = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return(pass);
|
|
|
|
} /* TestForCrucialController() */
|
|
|
|
void CrucialRegisterWrite(i2c_smbus_interface* bus, unsigned char dev, unsigned short reg, unsigned char val)
|
|
{
|
|
//Write Crucial register
|
|
bus->i2c_smbus_write_word_data(dev, 0x00, ((reg << 8) & 0xFF00) | ((reg >> 8) & 0x00FF));
|
|
|
|
//Write Crucial value
|
|
bus->i2c_smbus_write_byte_data(dev, 0x01, val);
|
|
}
|
|
|
|
/******************************************************************************************\
|
|
* *
|
|
* DetectCrucialControllers *
|
|
* *
|
|
* Detect Crucial controllers on the enumerated I2C busses. *
|
|
* *
|
|
* bus - pointer to i2c_smbus_interface where Aura device is connected *
|
|
* dev - I2C address of Aura device *
|
|
* *
|
|
\******************************************************************************************/
|
|
|
|
void DetectCrucialControllers(std::vector<i2c_smbus_interface*> &busses)
|
|
{
|
|
CrucialController* new_crucial;
|
|
RGBController_Crucial* new_controller;
|
|
|
|
for (unsigned int bus = 0; bus < busses.size(); bus++)
|
|
{
|
|
int address_list_idx = -1;
|
|
|
|
IF_DRAM_SMBUS(busses[bus]->pci_vendor, busses[bus]->pci_device)
|
|
{
|
|
|
|
for (unsigned int slot = 0; slot < 4; slot++)
|
|
{
|
|
int res = busses[bus]->i2c_smbus_write_quick(0x27, I2C_SMBUS_WRITE);
|
|
|
|
if (res < 0)
|
|
{
|
|
break;
|
|
}
|
|
|
|
LOG_DEBUG("[%s] Remapping RAM module on 0x27", CRUCIAL_CONTROLLER_NAME);
|
|
do
|
|
{
|
|
address_list_idx++;
|
|
|
|
if(address_list_idx < CRUCIAL_ADDRESS_COUNT)
|
|
{
|
|
res = busses[bus]->i2c_smbus_write_quick(crucial_addresses[address_list_idx], I2C_SMBUS_WRITE);
|
|
}
|
|
else
|
|
{
|
|
LOG_DEBUG("[%s] Testing address %02X to see if there is a device there", CRUCIAL_CONTROLLER_NAME, crucial_addresses[address_list_idx]);
|
|
break;
|
|
}
|
|
} while (res >= 0);
|
|
|
|
if(address_list_idx < CRUCIAL_ADDRESS_COUNT)
|
|
{
|
|
LOG_DEBUG("[%s] Remapping slot %d to address %02X", CRUCIAL_CONTROLLER_NAME, slot, crucial_addresses[address_list_idx]);
|
|
CrucialRegisterWrite(busses[bus], 0x27, 0x82EE, slot);
|
|
CrucialRegisterWrite(busses[bus], 0x27, 0x82EF, (crucial_addresses[address_list_idx] << 1));
|
|
CrucialRegisterWrite(busses[bus], 0x27, 0x82F0, 0xF0);
|
|
}
|
|
|
|
std::this_thread::sleep_for(1ms);
|
|
}
|
|
|
|
LOG_DEBUG("[%s] In bus: %02X:%02X looking for devices at [%s]", CRUCIAL_CONTROLLER_NAME, busses[bus]->pci_vendor, busses[bus]->pci_device, TESTING_ADDRESSES);
|
|
|
|
// Add Crucial controllers
|
|
for (unsigned int address_list_idx = 0; address_list_idx < CRUCIAL_ADDRESS_COUNT; address_list_idx++)
|
|
{
|
|
if (TestForCrucialController(busses[bus], crucial_addresses[address_list_idx]))
|
|
{
|
|
new_crucial = new CrucialController(busses[bus], crucial_addresses[address_list_idx]);
|
|
new_controller = new RGBController_Crucial(new_crucial);
|
|
ResourceManager::get()->RegisterRGBController(new_controller);
|
|
}
|
|
|
|
std::this_thread::sleep_for(1ms);
|
|
}
|
|
}
|
|
}
|
|
|
|
} /* DetectCrucialControllers() */
|
|
|
|
REGISTER_I2C_DETECTOR("Crucial", DetectCrucialControllers);
|