/* * This file is part of the coreboot project. * * Copyright 2013 Google Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2 of the License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include #include #include #include #include #include #include #if CONFIG_SOC_INTEL_BAYTRAIL #include #endif #define POLL_DELAY 100 /* 100us */ #if defined(__PRE_RAM__) #define EMPTY_DEV 0 #else #define EMPTY_DEV NULL #endif struct reg_script_context { device_t dev; struct resource *res; const struct reg_script *step; }; static inline void reg_script_set_dev(struct reg_script_context *ctx, device_t dev) { ctx->dev = dev; ctx->res = NULL; } static inline void reg_script_set_step(struct reg_script_context *ctx, const struct reg_script *step) { ctx->step = step; } static inline const struct reg_script * reg_script_get_step(struct reg_script_context *ctx) { return ctx->step; } static struct resource *reg_script_get_resource(struct reg_script_context *ctx) { #if defined(__PRE_RAM__) return NULL; #else struct resource *res; const struct reg_script *step = reg_script_get_step(ctx); res = ctx->res; if (res != NULL && res->index == step->res_index) return res; res = find_resource(ctx->dev, step->res_index); ctx->res = res; return res; #endif } static uint32_t reg_script_read_pci(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->size) { case REG_SCRIPT_SIZE_8: return pci_read_config8(ctx->dev, step->reg); case REG_SCRIPT_SIZE_16: return pci_read_config16(ctx->dev, step->reg); case REG_SCRIPT_SIZE_32: return pci_read_config32(ctx->dev, step->reg); } return 0; } static void reg_script_write_pci(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->size) { case REG_SCRIPT_SIZE_8: pci_write_config8(ctx->dev, step->reg, step->value); break; case REG_SCRIPT_SIZE_16: pci_write_config16(ctx->dev, step->reg, step->value); break; case REG_SCRIPT_SIZE_32: pci_write_config32(ctx->dev, step->reg, step->value); break; } } static uint32_t reg_script_read_io(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->size) { case REG_SCRIPT_SIZE_8: return inb(step->reg); case REG_SCRIPT_SIZE_16: return inw(step->reg); case REG_SCRIPT_SIZE_32: return inl(step->reg); } return 0; } static void reg_script_write_io(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->size) { case REG_SCRIPT_SIZE_8: outb(step->value, step->reg); break; case REG_SCRIPT_SIZE_16: outw(step->value, step->reg); break; case REG_SCRIPT_SIZE_32: outl(step->value, step->reg); break; } } static uint32_t reg_script_read_mmio(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->size) { case REG_SCRIPT_SIZE_8: return read8(step->reg); case REG_SCRIPT_SIZE_16: return read16(step->reg); case REG_SCRIPT_SIZE_32: return read32(step->reg); } return 0; } static void reg_script_write_mmio(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->size) { case REG_SCRIPT_SIZE_8: write8(step->reg, step->value); break; case REG_SCRIPT_SIZE_16: write16(step->reg, step->value); break; case REG_SCRIPT_SIZE_32: write32(step->reg, step->value); break; } } static uint32_t reg_script_read_res(struct reg_script_context *ctx) { struct resource *res; uint32_t val = 0; const struct reg_script *step = reg_script_get_step(ctx); res = reg_script_get_resource(ctx); if (res == NULL) return val; if (res->flags & IORESOURCE_IO) { const struct reg_script io_step = { .size = step->size, .reg = res->base + step->reg, }; reg_script_set_step(ctx, &io_step); val = reg_script_read_io(ctx); } else if (res->flags & IORESOURCE_MEM) { const struct reg_script mmio_step = { .size = step->size, .reg = res->base + step->reg, }; reg_script_set_step(ctx, &mmio_step); val = reg_script_read_mmio(ctx); } reg_script_set_step(ctx, step); return val; } static void reg_script_write_res(struct reg_script_context *ctx) { struct resource *res; const struct reg_script *step = reg_script_get_step(ctx); res = reg_script_get_resource(ctx); if (res == NULL) return; if (res->flags & IORESOURCE_IO) { const struct reg_script io_step = { .size = step->size, .reg = res->base + step->reg, .value = step->value, }; reg_script_set_step(ctx, &io_step); reg_script_write_io(ctx); } else if (res->flags & IORESOURCE_MEM) { const struct reg_script mmio_step = { .size = step->size, .reg = res->base + step->reg, .value = step->value, }; reg_script_set_step(ctx, &mmio_step); reg_script_write_mmio(ctx); } reg_script_set_step(ctx, step); } static uint32_t reg_script_read_iosf(struct reg_script_context *ctx) { #if CONFIG_SOC_INTEL_BAYTRAIL const struct reg_script *step = reg_script_get_step(ctx); switch (step->id) { case IOSF_PORT_BUNIT: return iosf_bunit_read(step->reg); case IOSF_PORT_DUNIT_CH0: return iosf_dunit_ch0_read(step->reg); case IOSF_PORT_PMC: return iosf_punit_read(step->reg); case IOSF_PORT_USBPHY: return iosf_usbphy_read(step->reg); case IOSF_PORT_USHPHY: return iosf_ushphy_read(step->reg); } #endif return 0; } static void reg_script_write_iosf(struct reg_script_context *ctx) { #if CONFIG_SOC_INTEL_BAYTRAIL const struct reg_script *step = reg_script_get_step(ctx); switch (step->id) { case IOSF_PORT_BUNIT: iosf_bunit_write(step->reg, step->value); break; case IOSF_PORT_DUNIT_CH0: iosf_dunit_write(step->reg, step->value); break; case IOSF_PORT_PMC: iosf_punit_write(step->reg, step->value); break; case IOSF_PORT_USBPHY: iosf_usbphy_write(step->reg, step->value); break; case IOSF_PORT_USHPHY: iosf_ushphy_write(step->reg, step->value); break; } #endif } static uint32_t reg_script_read(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->type) { case REG_SCRIPT_TYPE_PCI: return reg_script_read_pci(ctx); case REG_SCRIPT_TYPE_IO: return reg_script_read_io(ctx); case REG_SCRIPT_TYPE_MMIO: return reg_script_read_mmio(ctx); case REG_SCRIPT_TYPE_RES: return reg_script_read_res(ctx); case REG_SCRIPT_TYPE_IOSF: return reg_script_read_iosf(ctx); } return 0; } static void reg_script_write(struct reg_script_context *ctx) { const struct reg_script *step = reg_script_get_step(ctx); switch (step->type) { case REG_SCRIPT_TYPE_PCI: reg_script_write_pci(ctx); break; case REG_SCRIPT_TYPE_IO: reg_script_write_io(ctx); break; case REG_SCRIPT_TYPE_MMIO: reg_script_write_mmio(ctx); break; case REG_SCRIPT_TYPE_RES: reg_script_write_res(ctx); break; case REG_SCRIPT_TYPE_IOSF: reg_script_write_iosf(ctx); break; } } static void reg_script_rmw(struct reg_script_context *ctx) { uint32_t value; const struct reg_script *step = reg_script_get_step(ctx); struct reg_script write_step = *step; value = reg_script_read(ctx); value &= step->mask; value |= step->value; write_step.value = value; reg_script_set_step(ctx, &write_step); reg_script_write(ctx); reg_script_set_step(ctx, step); } /* In order to easily chain scripts together handle the REG_SCRIPT_COMMAND_NEXT * as recursive call with a new context that has the same dev and resource * as the previous one. That will run to completion and then move on to the * next step of the previous context. */ static void reg_script_run_next(struct reg_script_context *ctx, const struct reg_script *step); static void reg_script_run_with_context(struct reg_script_context *ctx) { uint32_t value = 0, try; while (1) { const struct reg_script *step = reg_script_get_step(ctx); if (step->command == REG_SCRIPT_COMMAND_END) break; switch (step->command) { case REG_SCRIPT_COMMAND_READ: (void)reg_script_read(ctx); break; case REG_SCRIPT_COMMAND_WRITE: reg_script_write(ctx); break; case REG_SCRIPT_COMMAND_RMW: reg_script_rmw(ctx); break; case REG_SCRIPT_COMMAND_POLL: for (try = 0; try < step->timeout; try += POLL_DELAY) { value = reg_script_read(ctx) & step->mask; if (value == step->value) break; udelay(POLL_DELAY); } if (try >= step->timeout) printk(BIOS_WARNING, "%s: POLL timeout waiting " "for 0x%08x to be 0x%08x, got 0x%08x\n", __func__, step->reg, step->value, value); break; case REG_SCRIPT_COMMAND_SET_DEV: reg_script_set_dev(ctx, step->dev); break; case REG_SCRIPT_COMMAND_NEXT: reg_script_run_next(ctx, step->next); break; default: printk(BIOS_WARNING, "Invalid command: %08x\n", step->command); break; } reg_script_set_step(ctx, step + 1); } } static void reg_script_run_next(struct reg_script_context *prev_ctx, const struct reg_script *step) { struct reg_script_context ctx; /* Use prev context as a basis but start at a new step. */ ctx = *prev_ctx; reg_script_set_step(&ctx, step); reg_script_run_with_context(&ctx); } void reg_script_run(const struct reg_script *step) { struct reg_script_context ctx; reg_script_set_dev(&ctx, EMPTY_DEV); reg_script_set_step(&ctx, step); reg_script_run_with_context(&ctx); }