iBoot/drivers/display_pmu/chestnut/chestnut.c

157 lines
3.8 KiB
C

/*
* Copyright (C) 2012-2013 Apple Inc. All rights reserved.
*
* This document is the property of Apple Inc.
* It is considered confidential and proprietary.
*
* This document may not be reproduced or transmitted in any form,
* in whole or in part, without the express written permission of
* Apple Inc.
*/
#include <debug.h>
#include <drivers/iic.h>
#include <drivers/display_pmu.h>
#include <lib/devicetree.h>
#include <target/gpiodef.h>
#include <target.h>
#define IIC_ADDRESS (0x4E)
#define DEVICE_ID_0 (0x00)
#define DEVICE_ID_1 (0x01)
#define DEVICE_ID_2 (0x02)
#define REVISION (0x03)
#define STATUS (0x04)
#define ENABLE (0x05)
#define VNEG_CONTROL (0x0F)
#define TI_DEVICE_ID0 (0x007365)
#define TI_DEVICE_ID1 (0x00F365)
#define INTERSIL_DEVICE_ID (0x0BA46E)
uint32_t chestnut_device_id;
uint8_t chestnut_device_revision;
bool chestnut_probed;
bool chestnut_present;
static const char *chestnut_device_string(uint32_t device_id)
{
switch (device_id) {
case TI_DEVICE_ID0:
case TI_DEVICE_ID1:
return "TI";
case INTERSIL_DEVICE_ID:
return "Intersil";
}
return "Unknown";
}
static void chestnut_probe()
{
if (!chestnut_probed) {
// Probe by reading a register
uint8_t data[4] = { 0 };
data[0] = REVISION;
if (iic_read(DISPLAY_PMU_IIC_BUS, IIC_ADDRESS, (const void *)data, 1, (void *)(data + 1), 1, IIC_NORMAL) == 0) {
chestnut_device_revision = data[1];
data[0] = DEVICE_ID_0;
if (iic_read(DISPLAY_PMU_IIC_BUS, IIC_ADDRESS, (const void *)data, 1, (void *)(data + 1), 3, IIC_NORMAL) == 0)
chestnut_device_id = (data[1] << 16) | (data[2] << 8) | data[3];
chestnut_present = true;
}
chestnut_probed = true;
}
if (chestnut_present)
dprintf(DEBUG_CRITICAL, "Display PMU found, 0x%08x, %s-%02x\n", chestnut_device_id, chestnut_device_string(chestnut_device_id), chestnut_device_revision);
else
dprintf(DEBUG_INFO, "Display PMU probe failed\n");
}
#if TARGET_CHESTNUT_USE_LOW_RIPPLE_MODE && !defined(TARGET_CHESTNUT_VNEG_CONTROL)
#define TARGET_CHESTNUT_VNEG_CONTROL 0x07
#endif
int display_pmu_init(void)
{
chestnut_probe();
if (chestnut_present) {
uint8_t data[2];
#if defined(TARGET_CHESTNUT_VNEG_CONTROL)
// Set low-ripple mode for VNEG
// <rdar://problem/12983634> N5x | Set low-ripple mode in Chestnut before Chestnut rails are enabled
data[0] = VNEG_CONTROL;
data[1] = TARGET_CHESTNUT_VNEG_CONTROL;
RELEASE_ASSERT(iic_write(DISPLAY_PMU_IIC_BUS, IIC_ADDRESS, data, 2) == 0);
#endif
// Enable power to the panel
data[0] = ENABLE;
data[1] = 0x3; // Enable Boost (VNEG and VPOS LDOs)
data[1] |= target_get_lcm_ldos();
RELEASE_ASSERT(iic_write(DISPLAY_PMU_IIC_BUS, IIC_ADDRESS, data, 2) == 0);
return 0;
}
return -1;
}
int display_pmu_quiesce(void)
{
if (chestnut_present) {
uint8_t data[2];
// Disable power to the panel
data[0] = ENABLE;
data[1] = 0;
RELEASE_ASSERT(iic_write(DISPLAY_PMU_IIC_BUS, IIC_ADDRESS, data, 2) == 0);
return 0;
}
return -1;
}
#if WITH_DEVICETREE
void display_pmu_update_device_tree(const char *node_string)
{
uint32_t propSize;
char *propName;
void *propData;
DTNode *node;
if (!chestnut_present)
return;
// Find display-pmu node
if (FindNode(0, node_string, &node)) {
// Fill in the probed device info
propName = "device-id";
if (FindProperty(node, &propName, &propData, &propSize))
if (propSize >= sizeof(chestnut_device_id))
*(uint32_t *)propData = chestnut_device_id;
propName = "device-revision";
if (FindProperty(node, &propName, &propData, &propSize))
if (propSize >= sizeof(chestnut_device_revision))
*(uint32_t *)propData = chestnut_device_revision;
propName = "device-string";
if (FindProperty(node, &propName, &propData, &propSize)) {
bzero(propData, propSize);
strlcpy(propData, chestnut_device_string(chestnut_device_id), propSize);
}
}
}
#endif