162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * kmsg dumper that ensures the OPAL console fully flushes panic messages
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Author: Russell Currey <ruscur@russell.cc>
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Copyright 2015 IBM Corporation.
862306a36Sopenharmony_ci */
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci#include <linux/kmsg_dump.h>
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ci#include <asm/opal.h>
1362306a36Sopenharmony_ci#include <asm/opal-api.h>
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_ci/*
1662306a36Sopenharmony_ci * Console output is controlled by OPAL firmware.  The kernel regularly calls
1762306a36Sopenharmony_ci * OPAL_POLL_EVENTS, which flushes some console output.  In a panic state,
1862306a36Sopenharmony_ci * however, the kernel no longer calls OPAL_POLL_EVENTS and the panic message
1962306a36Sopenharmony_ci * may not be completely printed.  This function does not actually dump the
2062306a36Sopenharmony_ci * message, it just ensures that OPAL completely flushes the console buffer.
2162306a36Sopenharmony_ci */
2262306a36Sopenharmony_cistatic void kmsg_dump_opal_console_flush(struct kmsg_dumper *dumper,
2362306a36Sopenharmony_ci				     enum kmsg_dump_reason reason)
2462306a36Sopenharmony_ci{
2562306a36Sopenharmony_ci	/*
2662306a36Sopenharmony_ci	 * Outside of a panic context the pollers will continue to run,
2762306a36Sopenharmony_ci	 * so we don't need to do any special flushing.
2862306a36Sopenharmony_ci	 */
2962306a36Sopenharmony_ci	if (reason != KMSG_DUMP_PANIC)
3062306a36Sopenharmony_ci		return;
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci	opal_flush_console(0);
3362306a36Sopenharmony_ci}
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_cistatic struct kmsg_dumper opal_kmsg_dumper = {
3662306a36Sopenharmony_ci	.dump = kmsg_dump_opal_console_flush
3762306a36Sopenharmony_ci};
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_civoid __init opal_kmsg_init(void)
4062306a36Sopenharmony_ci{
4162306a36Sopenharmony_ci	int rc;
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci	/* Add our dumper to the list */
4462306a36Sopenharmony_ci	rc = kmsg_dump_register(&opal_kmsg_dumper);
4562306a36Sopenharmony_ci	if (rc != 0)
4662306a36Sopenharmony_ci		pr_err("opal: kmsg_dump_register failed; returned %d\n", rc);
4762306a36Sopenharmony_ci}
48