Commit 5aff9b79 authored by He Zhe's avatar He Zhe Committed by Kleber Sacilotto de Souza

printk: Give error on attempt to set log buffer length to over 2G

BugLink: https://bugs.launchpad.net/bugs/1853881

[ Upstream commit e6fe3e5b ]

The current printk() is ready to handle log buffer size up to 2G.
Give an explicit error for users who want to use larger log buffer.

Also fix printk formatting to show the 2G as a positive number.

Link: http://lkml.kernel.org/r/20181008135916.gg4kkmoki5bgtco5@pathway.suse.cz
Cc: rostedt@goodmis.org
Cc: linux-kernel@vger.kernel.org
Suggested-by: default avatarSergey Senozhatsky <sergey.senozhatsky@gmail.com>
Signed-off-by: default avatarHe Zhe <zhe.he@windriver.com>
Reviewed-by: default avatarSergey Senozhatsky <sergey.senozhatsky@gmail.com>
[pmladek: Fixed to the really safe limit 2GB.]
Signed-off-by: default avatarPetr Mladek <pmladek@suse.com>
Signed-off-by: default avatarSasha Levin <sashal@kernel.org>
Signed-off-by: default avatarConnor Kuehl <connor.kuehl@canonical.com>
Signed-off-by: default avatarKhalid Elmously <khalid.elmously@canonical.com>
parent 26dea343
...@@ -279,6 +279,7 @@ static u32 clear_idx; ...@@ -279,6 +279,7 @@ static u32 clear_idx;
#define LOG_ALIGN __alignof__(struct printk_log) #define LOG_ALIGN __alignof__(struct printk_log)
#endif #endif
#define __LOG_BUF_LEN (1 << CONFIG_LOG_BUF_SHIFT) #define __LOG_BUF_LEN (1 << CONFIG_LOG_BUF_SHIFT)
#define LOG_BUF_LEN_MAX (u32)(1 << 31)
static char __log_buf[__LOG_BUF_LEN] __aligned(LOG_ALIGN); static char __log_buf[__LOG_BUF_LEN] __aligned(LOG_ALIGN);
static char *log_buf = __log_buf; static char *log_buf = __log_buf;
static u32 log_buf_len = __LOG_BUF_LEN; static u32 log_buf_len = __LOG_BUF_LEN;
...@@ -870,18 +871,23 @@ void log_buf_kexec_setup(void) ...@@ -870,18 +871,23 @@ void log_buf_kexec_setup(void)
static unsigned long __initdata new_log_buf_len; static unsigned long __initdata new_log_buf_len;
/* we practice scaling the ring buffer by powers of 2 */ /* we practice scaling the ring buffer by powers of 2 */
static void __init log_buf_len_update(unsigned size) static void __init log_buf_len_update(u64 size)
{ {
if (size > (u64)LOG_BUF_LEN_MAX) {
size = (u64)LOG_BUF_LEN_MAX;
pr_err("log_buf over 2G is not supported.\n");
}
if (size) if (size)
size = roundup_pow_of_two(size); size = roundup_pow_of_two(size);
if (size > log_buf_len) if (size > log_buf_len)
new_log_buf_len = size; new_log_buf_len = (unsigned long)size;
} }
/* save requested log_buf_len since it's too early to process it */ /* save requested log_buf_len since it's too early to process it */
static int __init log_buf_len_setup(char *str) static int __init log_buf_len_setup(char *str)
{ {
unsigned int size; u64 size;
if (!str) if (!str)
return -EINVAL; return -EINVAL;
...@@ -951,7 +957,7 @@ void __init setup_log_buf(int early) ...@@ -951,7 +957,7 @@ void __init setup_log_buf(int early)
} }
if (unlikely(!new_log_buf)) { if (unlikely(!new_log_buf)) {
pr_err("log_buf_len: %ld bytes not available\n", pr_err("log_buf_len: %lu bytes not available\n",
new_log_buf_len); new_log_buf_len);
return; return;
} }
...@@ -964,8 +970,8 @@ void __init setup_log_buf(int early) ...@@ -964,8 +970,8 @@ void __init setup_log_buf(int early)
memcpy(log_buf, __log_buf, __LOG_BUF_LEN); memcpy(log_buf, __log_buf, __LOG_BUF_LEN);
raw_spin_unlock_irqrestore(&logbuf_lock, flags); raw_spin_unlock_irqrestore(&logbuf_lock, flags);
pr_info("log_buf_len: %d bytes\n", log_buf_len); pr_info("log_buf_len: %u bytes\n", log_buf_len);
pr_info("early log buf free: %d(%d%%)\n", pr_info("early log buf free: %u(%u%%)\n",
free, (free * 100) / __LOG_BUF_LEN); free, (free * 100) / __LOG_BUF_LEN);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment