1 // SPDX-License-Identifier: GPL-2.0
3 * Copyright (c) 2016 Google, Inc
5 * Modified from coreboot
10 #include <asm/intel_regs.h>
12 #include <asm/arch/pch.h>
14 #define IOBP_RETRY 1000
16 /* IO Buffer Programming */
17 #define IOBPIRI 0x2330
20 #define IOBPS_READY 0x0001
21 #define IOBPS_TX_MASK 0x0006
22 #define IOBPS_MASK 0xff00
23 #define IOBPS_READ 0x0600
24 #define IOBPS_WRITE 0x0700
26 #define IOBPU_MAGIC 0xf000
27 #define IOBP_PCICFG_READ 0x0400
28 #define IOBP_PCICFG_WRITE 0x0500
30 static inline int iobp_poll(void)
34 for (try = IOBP_RETRY; try > 0; try--) {
35 u16 status = readw(RCB_REG(IOBPS));
36 if ((status & IOBPS_READY) == 0)
41 printf("IOBP: timeout waiting for transaction to complete\n");
45 int pch_iobp_trans_start(u32 address, int op)
51 writel(address, RCB_REG(IOBPIRI));
54 clrsetbits_le16(RCB_REG(IOBPS), IOBPS_MASK, op);
59 int pch_iobp_trans_finish(void)
63 /* Undocumented magic */
64 writew(IOBPU_MAGIC, RCB_REG(IOBPU));
67 setbits_le16(RCB_REG(IOBPS), IOBPS_READY);
72 /* Check for successful transaction */
73 status = readw(RCB_REG(IOBPS));
74 if (status & IOBPS_TX_MASK)
80 u32 pch_iobp_read(u32 address)
82 if (!pch_iobp_trans_start(address, IOBPS_READ))
84 if (pch_iobp_trans_finish()) {
85 printf("IOBP: read 0x%08x failed\n", address);
90 return readl(RCB_REG(IOBPD));
93 int pch_iobp_write(u32 address, u32 data)
95 if (!pch_iobp_trans_start(address, IOBPS_WRITE))
98 writel(data, RCB_REG(IOBPD));
100 if (pch_iobp_trans_finish()) {
101 printf("IOBP: write 0x%08x failed\n", address);
108 int pch_iobp_update(u32 address, u32 andvalue, u32 orvalue)
110 u32 data = pch_iobp_read(address);
112 /* Update the data */
116 return pch_iobp_write(address, data);
119 int pch_iobp_exec(u32 addr, u16 op_code, u8 route_id, u32 *data, u8 *resp)
128 writel(addr, RCB_REG(IOBPIRI));
129 clrsetbits_le16(RCB_REG(IOBPS), 0xff00, op_code);
130 writew(IOBPU_MAGIC | route_id, RCB_REG(IOBPU));
132 writel(*data, RCB_REG(IOBPD));
133 /* Set IOBPS[0] to trigger IOBP transaction*/
134 setbits_le16(RCB_REG(IOBPS), 1);
139 *resp = (readw(RCB_REG(IOBPS)) & IOBPS_TX_MASK) >> 1;
140 *data = readl(RCB_REG(IOBPD));