| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * (C) Copyright 2018 |
| * Mario Six, Guntermann & Drunck GmbH, mario.six@gdsys.cc |
| */ |
| |
| #include <common.h> |
| #include <command.h> |
| #include <dm.h> |
| #include <log.h> |
| #include <sysreset.h> |
| #include <wait_bit.h> |
| #include <linux/delay.h> |
| |
| #include "sysreset_mpc83xx.h" |
| |
| /* Magic 4-byte word to enable reset ('RSTE' in ASCII) */ |
| static const u32 RPR_MAGIC = 0x52535445; |
| /* Wait at most 2000ms for reset control enable bit */ |
| static const uint RESET_WAIT_TIMEOUT = 2000; |
| |
| /** |
| * __do_reset() - Execute the system reset |
| * |
| * Return: The functions resets the system, and never returns. |
| */ |
| static int __do_reset(void) |
| { |
| ulong msr; |
| int res; |
| |
| immap_t *immap = (immap_t *)CONFIG_SYS_IMMR; |
| |
| puts("Resetting the board.\n"); |
| |
| /* Interrupts and MMU off */ |
| msr = mfmsr(); |
| msr &= ~(MSR_EE | MSR_IR | MSR_DR); |
| mtmsr(msr); |
| |
| /* Enable Reset Control Reg */ |
| out_be32(&immap->reset.rpr, RPR_MAGIC); |
| sync(); |
| isync(); |
| |
| /* Confirm Reset Control Reg is enabled */ |
| res = wait_for_bit_be32(&immap->reset.rcer, RCER_CRE, true, |
| RESET_WAIT_TIMEOUT, false); |
| if (res) { |
| debug("%s: Timed out waiting for reset control to be set\n", |
| __func__); |
| return res; |
| } |
| |
| udelay(200); |
| |
| /* Perform reset, only one bit */ |
| out_be32(&immap->reset.rcr, RCR_SWHR); |
| |
| /* Never executes */ |
| return 0; |
| } |
| |
| static int mpc83xx_sysreset_request(struct udevice *dev, enum sysreset_t type) |
| { |
| switch (type) { |
| case SYSRESET_WARM: |
| case SYSRESET_COLD: |
| return __do_reset(); |
| default: |
| return -EPROTONOSUPPORT; |
| } |
| |
| return -EINPROGRESS; |
| } |
| |
| /** |
| * print_83xx_arb_event() - Print arbiter events to buffer |
| * @force: Print arbiter events, even if none are indicated by the system |
| * @buf: The buffer to receive the printed arbiter event information |
| * @size: The size of the buffer to receive the printed arbiter event |
| * information in bytes |
| * |
| * Return: Number of bytes printed to buffer, -ve on error |
| */ |
| static int print_83xx_arb_event(bool force, char *buf, int size) |
| { |
| int etype = (gd->arch.arbiter_event_attributes & AEATR_EVENT) |
| >> AEATR_EVENT_SHIFT; |
| int mstr_id = (gd->arch.arbiter_event_attributes & AEATR_MSTR_ID) |
| >> AEATR_MSTR_ID_SHIFT; |
| int tbst = (gd->arch.arbiter_event_attributes & AEATR_TBST) |
| >> AEATR_TBST_SHIFT; |
| int tsize = (gd->arch.arbiter_event_attributes & AEATR_TSIZE) |
| >> AEATR_TSIZE_SHIFT; |
| int ttype = (gd->arch.arbiter_event_attributes & AEATR_TTYPE) |
| >> AEATR_TTYPE_SHIFT; |
| int tsize_val = (tbst << 3) | tsize; |
| int tsize_bytes = tbst ? (tsize ? tsize : 8) : 16 + 8 * tsize; |
| int res = 0; |
| |
| /* |
| * If we don't force output, and there is no event (event address == |
| * 0), then don't print anything |
| */ |
| if (!force && !gd->arch.arbiter_event_address) |
| return 0; |
| |
| if (CONFIG_IS_ENABLED(DISPLAY_AER_FULL)) { |
| res = snprintf(buf, size, |
| "Arbiter Event Status:\n" |
| " %s: 0x%08lX\n" |
| " %s: 0x%1x = %s\n" |
| " %s: 0x%02x = %s\n" |
| " %s: 0x%1x = %d bytes\n" |
| " %s: 0x%02x = %s\n", |
| "Event Address", gd->arch.arbiter_event_address, |
| "Event Type", etype, event[etype], |
| "Master ID", mstr_id, master[mstr_id], |
| "Transfer Size", tsize_val, tsize_bytes, |
| "Transfer Type", ttype, transfer[ttype]); |
| } else if (CONFIG_IS_ENABLED(DISPLAY_AER_BRIEF)) { |
| res = snprintf(buf, size, |
| "Arbiter Event Status: AEATR=0x%08lX, AEADR=0x%08lX\n", |
| gd->arch.arbiter_event_attributes, |
| gd->arch.arbiter_event_address); |
| } |
| |
| return res; |
| } |
| |
| static int mpc83xx_sysreset_get_status(struct udevice *dev, char *buf, int size) |
| { |
| /* Ad-hoc data structure to map RSR bit values to their descriptions */ |
| static const struct { |
| /* Bit mask for the bit in question */ |
| ulong mask; |
| /* Description of the bitmask in question */ |
| char *desc; |
| } bits[] = { |
| { |
| RSR_SWSR, "Software Soft"}, { |
| RSR_SWHR, "Software Hard"}, { |
| RSR_JSRS, "JTAG Soft"}, { |
| RSR_CSHR, "Check Stop"}, { |
| RSR_SWRS, "Software Watchdog"}, { |
| RSR_BMRS, "Bus Monitor"}, { |
| RSR_SRS, "External/Internal Soft"}, { |
| RSR_HRS, "External/Internal Hard"} |
| }; |
| int res; |
| ulong rsr = gd->arch.reset_status; |
| int i; |
| char *sep; |
| |
| res = snprintf(buf, size, "Reset Status:"); |
| if (res < 0) { |
| debug("%s: Could not write reset status message (err = %d)\n", |
| dev->name, res); |
| return -EIO; |
| } |
| |
| buf += res; |
| size -= res; |
| |
| sep = " "; |
| for (i = 0; i < ARRAY_SIZE(bits); i++) |
| /* Print description of set bits */ |
| if (rsr & bits[i].mask) { |
| res = snprintf(buf, size, "%s%s%s", sep, bits[i].desc, |
| (i == ARRAY_SIZE(bits) - 1) ? "\n" : ""); |
| if (res < 0) { |
| debug("%s: Could not write reset status message (err = %d)\n", |
| dev->name, res); |
| return -EIO; |
| } |
| buf += res; |
| size -= res; |
| sep = ", "; |
| } |
| |
| /* |
| * TODO(mario.six@gdsys.cc): Move this into a dedicated |
| * arbiter driver |
| */ |
| if (CONFIG_IS_ENABLED(DISPLAY_AER_FULL) || |
| CONFIG_IS_ENABLED(DISPLAY_AER_BRIEF)) { |
| /* |
| * If there was a bus monitor reset event, we force the arbiter |
| * event to be printed |
| */ |
| res = print_83xx_arb_event(rsr & RSR_BMRS, buf, size); |
| if (res < 0) { |
| debug("%s: Could not write arbiter event message (err = %d)\n", |
| dev->name, res); |
| return -EIO; |
| } |
| buf += res; |
| size -= res; |
| } |
| snprintf(buf, size, "\n"); |
| |
| return 0; |
| } |
| |
| static struct sysreset_ops mpc83xx_sysreset = { |
| .request = mpc83xx_sysreset_request, |
| .get_status = mpc83xx_sysreset_get_status, |
| }; |
| |
| U_BOOT_DRIVER(sysreset_mpc83xx) = { |
| .name = "mpc83xx_sysreset", |
| .id = UCLASS_SYSRESET, |
| .ops = &mpc83xx_sysreset, |
| }; |