[PATCH] Fix a segfault by "net" command
by Kazuhito Hagio
Hi Dave,
When a network device has a lot of IP addresses, the "net" command
can generate a segmentation fault due to a buffer overflow without
this patch. I have seen several vmcores like that in customer support.
# for i in {1..250} ; do ifconfig eth1:$i 192.168.1.$i ; done
# crash
...
crash> net
NET_DEVICE NAME IP ADDRESS(ES)
ffff88007faab000 lo 127.0.0.1
ffff88003e097000 eth0 192.168.122.182
ffff88003e12b000 eth1 192.168.1.1, 192.168.1.2, ..., 192.168.1.250
ffff88003e12e000 eth2
Segmentation fault (core dumped)
Signed-off-by: Kazuhito Hagio <k-hagio(a)ab.jp.nec.com>
---
net.c | 48 +++++++++++++++++++++++++++++++++++-------------
1 file changed, 35 insertions(+), 13 deletions(-)
diff --git a/net.c b/net.c
index bb86963..7b70dc4 100644
--- a/net.c
+++ b/net.c
@@ -70,7 +70,7 @@ static void show_net_devices_v3(ulong);
static void print_neighbour_q(ulong, int);
static void get_netdev_info(ulong, struct devinfo *);
static void get_device_name(ulong, char *);
-static void get_device_address(ulong, char *);
+static long get_device_address(ulong, char **, long);
static void get_sock_info(ulong, char *);
static void dump_arp(void);
static void arp_state_to_flags(unsigned char);
@@ -441,7 +441,8 @@ show_net_devices(ulong task)
{
ulong next;
long flen;
- char buf[BUFSIZE];
+ char *buf;
+ long buflen = BUFSIZE;
if (symbol_exists("dev_base_head")) {
show_net_devices_v2(task);
@@ -459,6 +460,7 @@ show_net_devices(ulong task)
if (!net->netdevice || !next)
return;
+ buf = GETBUF(buflen);
flen = MAX(VADDR_PRLEN, strlen(net->netdevice));
fprintf(fp, "%s NAME IP ADDRESS(ES)\n",
@@ -472,12 +474,14 @@ show_net_devices(ulong task)
get_device_name(next, buf);
fprintf(fp, "%-6s ", buf);
- get_device_address(next, buf);
+ buflen = get_device_address(next, &buf, buflen);
fprintf(fp, "%s\n", buf);
readmem(next+net->dev_next, KVADDR, &next,
sizeof(void *), "(net_)device.next", FAULT_ON_ERROR);
} while (next);
+
+ FREEBUF(buf);
}
static void
@@ -485,13 +489,15 @@ show_net_devices_v2(ulong task)
{
struct list_data list_data, *ld;
char *net_device_buf;
- char buf[BUFSIZE];
+ char *buf;
+ long buflen = BUFSIZE;
int ndevcnt, i;
long flen;
if (!net->netdevice) /* initialized in net_init() */
return;
+ buf = GETBUF(buflen);
flen = MAX(VADDR_PRLEN, strlen(net->netdevice));
fprintf(fp, "%s NAME IP ADDRESS(ES)\n",
@@ -521,12 +527,13 @@ show_net_devices_v2(ulong task)
get_device_name(ld->list_ptr[i], buf);
fprintf(fp, "%-6s ", buf);
- get_device_address(ld->list_ptr[i], buf);
+ buflen = get_device_address(ld->list_ptr[i], &buf, buflen);
fprintf(fp, "%s\n", buf);
}
FREEBUF(ld->list_ptr);
FREEBUF(net_device_buf);
+ FREEBUF(buf);
}
static void
@@ -535,13 +542,15 @@ show_net_devices_v3(ulong task)
ulong nsproxy_p, net_ns_p;
struct list_data list_data, *ld;
char *net_device_buf;
- char buf[BUFSIZE];
+ char *buf;
+ long buflen = BUFSIZE;
int ndevcnt, i;
long flen;
if (!net->netdevice) /* initialized in net_init() */
return;
+ buf = GETBUF(buflen);
flen = MAX(VADDR_PRLEN, strlen(net->netdevice));
fprintf(fp, "%s NAME IP ADDRESS(ES)\n",
@@ -581,12 +590,13 @@ show_net_devices_v3(ulong task)
get_device_name(ld->list_ptr[i], buf);
fprintf(fp, "%-6s ", buf);
- get_device_address(ld->list_ptr[i], buf);
+ buflen = get_device_address(ld->list_ptr[i], &buf, buflen);
fprintf(fp, "%s\n", buf);
}
FREEBUF(ld->list_ptr);
FREEBUF(net_device_buf);
+ FREEBUF(buf);
}
/*
@@ -869,13 +879,18 @@ get_device_name(ulong devaddr, char *buf)
* in_ifaddr->ifa_next points to the next in_ifaddr in the list (if any).
*
*/
-static void
-get_device_address(ulong devaddr, char *buf)
+static long
+get_device_address(ulong devaddr, char **bufp, long buflen)
{
ulong ip_ptr, ifa_list;
struct in_addr ifa_address;
+ char *buf;
+ char buf2[BUFSIZE];
+ long pos = 0;
- BZERO(buf, BUFSIZE);
+ buf = *bufp;
+ BZERO(buf, buflen);
+ BZERO(buf2, BUFSIZE);
readmem(devaddr + net->dev_ip_ptr, KVADDR,
&ip_ptr, sizeof(ulong), "ip_ptr", FAULT_ON_ERROR);
@@ -891,13 +906,20 @@ get_device_address(ulong devaddr, char *buf)
&ifa_address, sizeof(struct in_addr), "ifa_address",
FAULT_ON_ERROR);
- sprintf(&buf[strlen(buf)], "%s%s",
- strlen(buf) ? ", " : "",
- inet_ntoa(ifa_address));
+ sprintf(buf2, "%s%s", pos ? ", " : "", inet_ntoa(ifa_address));
+ if (pos + strlen(buf2) >= buflen) {
+ RESIZEBUF(*bufp, buflen, buflen * 2);
+ buf = *bufp;
+ BZERO(buf + buflen, buflen);
+ buflen *= 2;
+ }
+ BCOPY(buf2, &buf[pos], strlen(buf2));
+ pos += strlen(buf2);
readmem(ifa_list + OFFSET(in_ifaddr_ifa_next), KVADDR,
&ifa_list, sizeof(ulong), "ifa_next", FAULT_ON_ERROR);
}
+ return buflen;
}
/*
--
1.8.3.1
7 years, 1 month
HEADS UP: upstream kernel pidhash removal
by Dave Anderson
FYI, if you're running with bleeding edge upstream kernels, this
recent commit destroys a fundamental underpinning required by the
crash utility:
commit e8cfbc245e24887e3c30235f71e9e9405e0cfc39
Author: Gargi Sharma <gs051095(a)gmail.com>
Date: Fri Nov 17 15:30:34 2017 -0800
pid: remove pidhash
pidhash is no longer required as all the information can be looked up
from idr tree. nr_hashed represented the number of pids that had been
hashed. Since, nr_hashed and PIDNS_HASH_ADDING are no longer relevant,
it has been renamed to pid_allocated and PIDNS_ADDING respectively.
I'm not sure how the failure will manifest itself, but it certainly will
will fail early on during session initialization. When I saw the original
patch postings, I started tinkering with the creation of yet another
function to plug into "tt->refresh_task_table" for the new IDR/radix-tree
scheme, but until I see an actual live kernel and/or vmcore with the patch
above, I've held off on further work on it.
In the meantime, you can use the "crash --active" command line option,
which restricts the task list to those that are registered as the current
task in each per-cpu runqueue.
Dave
7 years, 1 month
crash: page excluded: kernel virtual address: ffffffff81e3da50 type: "page_offset_base"
by Cao jin
Hi,
I am using the latest crash tool & kernel 4.14 compiled from source, and
I got the following error message. As I searched, this is fixed in crash
7.2.0, but I still have it here. So, is anyone has a clue?
[root@IAAS1 crash]# ./crash
/var/crash/127.0.0.1-2017-11-22-11\:57\:51/vmcore ../linux/vmlinux
crash 7.2.0++
Copyright (C) 2002-2017 Red Hat, Inc.
Copyright (C) 2004, 2005, 2006, 2010 IBM Corporation
Copyright (C) 1999-2006 Hewlett-Packard Co
Copyright (C) 2005, 2006, 2011, 2012 Fujitsu Limited
Copyright (C) 2006, 2007 VA Linux Systems Japan K.K.
Copyright (C) 2005, 2011 NEC Corporation
Copyright (C) 1999, 2002, 2007 Silicon Graphics, Inc.
Copyright (C) 1999, 2000, 2001, 2002 Mission Critical Linux, Inc.
This program is free software, covered by the GNU General Public License,
and you are welcome to change it and/or distribute copies of it under
certain conditions. Enter "help copying" to see the conditions.
This program has absolutely no warranty. Enter "help warranty" for details.
GNU gdb (GDB) 7.6
Copyright (C) 2013 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later
<http://gnu.org/licenses/gpl.html>
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law. Type "show copying"
and "show warranty" for details.
This GDB was configured as "x86_64-unknown-linux-gnu"...
crash: page excluded: kernel virtual address: ffffffff81e3da50 type:
"page_offset_base"
--
Sincerely,
Cao jin
7 years, 1 month
[PATCH RFCv1] Crash: Extensions: Coresight panic dump
by Leo Yan
This commit is to add support Coresight panic dump, it uses coresight
dump list head pointer to iterate coresight devices; if the device is
etm device then use the dump as meta data and use etf device as trace
data. It also generates 'perf' format compatible file so the trace
data can be analyzed by 'perf' tool.
This initial version has limitation:
- It can only generate data for etm and etf; haven't verified for other
coresight topology;
- It can only support etm/etf for N:1 relationship, haven't verified for
the device with multiple sink devices.
The building and usage are simple, for building can place csdump.c file
into 'extensions' folder, then use 'make extensions' to build code,
we can get one file name csdump.so.
Then we can use csdump.so in 'crash' tool to extract trace files:
crash> extend csdump.so
crash> csdump trace_out_dir
Can use 'extend -u' command to unload the module:
crash> extend -u csdump.so
Signed-off-by: Leo Yan <leo.yan(a)linaro.org>
---
extensions/csdump.c | 547 ++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 547 insertions(+)
create mode 100644 extensions/csdump.c
diff --git a/extensions/csdump.c b/extensions/csdump.c
new file mode 100644
index 0000000..c657378
--- /dev/null
+++ b/extensions/csdump.c
@@ -0,0 +1,547 @@
+/*
+ * Extension module to dump log buffer of ARM Coresight Trace
+ *
+ * Copyright (C) 2017 Linaro Ltd.
+ *
+ * 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; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ */
+
+#define _GNU_SOURCE
+#include <sys/file.h>
+
+#include "defs.h"
+
+static unsigned int perf_header[] = {
+ 0x46524550, 0x32454c49, /* Magic: PERFILE2 */
+ 0x00000068, 0x00000000, /* header size */
+ 0x00000080, 0x00000000, /* attr size */
+ 0x00000078, 0x00000000, /* attrs offset */
+ 0x00000100, 0x00000000, /* attrs size */
+ 0x00000178, 0x00000000, /* data offset */
+ 0x00002568, 0x00000000, /* data size */
+ 0x00000000, 0x00000000, /* event offset */
+ 0x00000000, 0x00000000, /* event size */
+ 0x00040004, 0x00000000, /* feature bitmap */
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000
+};
+
+static unsigned int perf_event_id[] = {
+ 0x00000015, 0x00000000,
+ 0x00000016, 0x00000000,
+};
+
+static unsigned int perf_event_cs_etm[] = {
+ 0x00000006, 0x00000070, /* event type: 6, size: 0x70 */
+ 0x00000000, 0x00000000, /* config: 0 */
+ 0x00000001, 0x00000000, /* sample_period: 1 */
+ 0x00010003, 0x00000000, /* sample_type: PERF_SAMPLE_IP | PERF_SAMPLE_TID |
+ PERF_SAMPLE_PERIOD */
+ 0x00000004, 0x00000000, /* read_format: PERF_FORMAT_ID */
+ 0x00141001, 0x00000000, /* disabled: 1, enable_on_exec: 1,
+ sample_id_all: 1, exclude_guest: 1 */
+ 0x00000000, 0x00000000, /* wakeup_events: 0, bp_type: 0 */
+ 0x00000000, 0x00000000, /* config1 : 0 */
+ 0x00000000, 0x00000000, /* config1 : 0 */
+ 0x00000000, 0x00000000, /* branch_sample_type: 0 */
+ 0x00000000, 0x00000000, /* sample_regs_user: 0 */
+ 0x00000000, 0x00000000, /* sample_stack_user: 0, clockid: 0 */
+ 0x00000000, 0x00000000, /* sample_regs_intr: 0 */
+ 0x00000000, 0x00000000, /* aux_watermark: 0, sample_max_stack: 0 */
+ 0x00000068, 0x00000000, /* ids.offset: 0x68 */
+ 0x00000008, 0x00000000, /* ids.size: 0x8 */
+};
+
+static unsigned int perf_event_dummy[] = {
+ 0x00000001, 0x00000070, /* event type: 1, size: 0x70 */
+ 0x00000009, 0x00000000, /* config: 9 */
+ 0x00000001, 0x00000000, /* sample_period: 1 */
+ 0x00010003, 0x00000000, /* sample_type: PERF_SAMPLE_IP | PERF_SAMPLE_TID |
+ PERF_SAMPLE_PERIOD */
+ 0x00000004, 0x00000000, /* read_format: PERF_FORMAT_ID */
+ 0x01843361, 0x00000000, /* disabled: 1, exclude_kernel: 1
+ exclude_hv: 1, mmap: 1,
+ comm: 1, enable_on_exec: 1,
+ task: 1, sample_id_all: 1,
+ mmap2: 1, comm_exec: 1 */
+ 0x00000000, 0x00000000, /* wakeup_events: 0, bp_type: 0 */
+ 0x00000000, 0x00000000, /* config1 : 0 */
+ 0x00000000, 0x00000000, /* config1 : 0 */
+ 0x00000000, 0x00000000, /* branch_sample_type: 0 */
+ 0x00000000, 0x00000000, /* sample_regs_user: 0 */
+ 0x00000000, 0x00000000, /* sample_stack_user: 0, clockid: 0 */
+ 0x00000000, 0x00000000, /* sample_regs_intr: 0 */
+ 0x00000000, 0x00000000, /* aux_watermark: 0, sample_max_stack: 0 */
+ 0x00000070, 0x00000000, /* ids.offset: 0x70 */
+ 0x00000008, 0x00000000, /* ids.size: 0x8 */
+};
+
+static unsigned int perf_auxtrace_info[] = {
+ 0x00000046, 0x02680000, /* type: PERF_RECORD_AUXTRACE_INFO, size: 0x268 */
+ 0x00000003, 0x00000000, /* info->type: PERF_AUXTRACE_CS_ETM */
+ 0x00000000, 0x00000000, /* version: 0 */
+ 0x00000008, 0x00000006, /* cpus: 8, type: 6 */
+ 0x00000000, 0x00000000 /* snapshot_mode: 0 */
+};
+
+static unsigned int perf_kernel_mmap[] = {
+ 0x00000001, 0x00500001, /* type: PERF_RECORD_MMAP, size: 0x50,
+ misc: PERF_RECORD_MISC_KERNEL */
+ 0xffffffff, 0x00000000, /* pid: 0xffffffff, tid: 0x0 */
+ 0x08080000, 0xffff0000, /* start: 0xffff000008080000 */
+ 0xf7f7ffff, 0x0000ffff, /* len: 0x0000fffff7f7ffff */
+ 0x08080000, 0xffff0000, /* pgoff: 0xffff000008080000 */
+ 0x72656b5b, 0x2e6c656e, /* filename: [kernel.kallsyms]_text */
+ 0x6c6c616b, 0x736d7973,
+ 0x65745f5d, 0x00007478,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+};
+
+static unsigned int perf_threads[] = {
+ 0x00000003, 0x00280000, /* type: PERF_RECORD_COMM, size: 0x28 */
+ 0x0000090c, 0x0000090c, /* pid: 0x90c, tid: 0x90c */
+ 0x66726570, 0x00000000, /* comm: perf */
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+
+ 0x00000003, 0x00282000, /* type: PERF_RECORD_COMM, size: 0x28 */
+ 0x0000090c, 0x0000090c, /* pid: 0x90c, tid: 0x90c */
+ 0x696e6170, 0x00000063, /* comm: panic */
+ 0x0000090c, 0x0000090c,
+ 0x00000016, 0x00000000,
+
+ 0x0000000a, 0x00680002, /* type: PERF_RECORD_MMAP2, size: 0x68 */
+ 0x0000090c, 0x0000090c, /* pid: 0x90c, tid: 0x90c */
+ 0x00400000, 0x00000000, /* addr: 0x00400000 */
+ 0x00006000, 0x00000000, /* len: 0x00006000 */
+ 0x00000000, 0x00000000, /* pgoff: 0x0 */
+ 0x000000b3, 0x00000009, /* maj: 0xb3, min: 0x9 */
+ 0x00000085, 0x00000000, /* ino: 0x85 */
+ 0x00000000, 0x00000000, /* ino_generation: 0x0 */
+ 0x00000005, 0x00001802, /* prot: PROT_READ | PROT_EXEC, flag: 0x1802 */
+ 0x6e69622f, 0x616e752f, /* comm: /bin/uname */
+ 0x0000656d, 0x00000000,
+ 0x0000090c, 0x0000090c,
+ 0x00000016, 0x00000000,
+
+ 0x0000000a, 0x00800002,
+ 0x0000090c, 0x0000090c,
+ 0xb77c2000, 0x0000ffff,
+ 0x0002e000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x000000b3, 0x00000009,
+ 0x00000752, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000005, 0x00001802,
+ 0x62696c2f, 0x7261612f, /* ld-2.19.so */
+ 0x34366863, 0x6e696c2d,
+ 0x672d7875, 0x6c2f756e,
+ 0x2e322d64, 0x732e3931,
+ 0x0000006f, 0x00000000,
+ 0x0000090c, 0x0000090c,
+ 0x00000016, 0x00000000,
+
+ 0x0000000a, 0x00600002,
+ 0x0000090c, 0x0000090c,
+ 0xb77ec000, 0x0000ffff,
+ 0x00001000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000005, 0x00001002,
+ 0x7364765b, 0x00005d6f, /* [vdso] */
+ 0x0000090c, 0x0000090c,
+ 0x00000016, 0x00000000,
+
+ 0x0000000a, 0x00800002,
+ 0x0000090c, 0x0000090c,
+ 0xb7675000, 0x0000ffff,
+ 0x0014d000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x000000b3, 0x00000009,
+ 0x0000076a, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000005, 0x00001002,
+ 0x62696c2f, 0x7261612f, /* /lib/aarch64-linux-gnu/libc-2.19.so */
+ 0x34366863, 0x6e696c2d,
+ 0x672d7875, 0x6c2f756e,
+ 0x2d636269, 0x39312e32,
+ 0x006f732e, 0x00000000,
+ 0x0000090c, 0x0000090c,
+ 0x00000016, 0x00000000,
+
+ 0x0000000b, 0x00300000, /* type: PERF_RECORD_AUX, size: 0x30 */
+ 0x00000000, 0x00000000, /* aux_offset: 0x0 */
+ 0x00002000, 0x00000000, /* aux_size: 0x2000 */
+ 0x00000001, 0x00000000, /* flag: 0x1 */
+ 0x0000090c, 0x0000090c,
+ 0x00000015, 0x00000000,
+
+ 0x00000004, 0x00300000, /* type: PERF_RECORD_EXIT, size: 0x30 */
+ 0x0000090c, 0x0000090c, /* pid, ppid: 0x90c */
+ 0x0000090c, 0x0000090c, /* tid, ptid: 0x90c */
+ 0xf89cbddc, 0x00000571,
+ 0x0000090c, 0x0000090c,
+ 0x00000016, 0x00000000,
+};
+
+static unsigned int perf_auxtrace_snapshot[] = {
+ 0x00000047, 0x00300000, /* type: PERF_RECORD_AUXTRACE, size: 0x30 */
+ 0x00002000, 0x00000000, /* auxsize: 0x2000 */
+ 0x00000000, 0x00000000, /* auxoffset: 0x0 */
+ 0x74bc6ab4, 0x5a1c47b3, /* reference: rand() */
+ 0x00000000, 0x0000090c, /* idx: 0x0, pid: 0x90c */
+ 0xffffffff, 0x00000000, /* cpu: -1 */
+};
+
+static unsigned int perf_record_finish[] = {
+ 0x00000044, 0x00080000,
+};
+
+static unsigned int perf_sections[] = {
+ 0x00002760, 0x00000000, /* buildid section: start addr */
+ 0x00000064, 0x00000000, /* buildid section: len */
+ 0x000027c4, 0x00000000, /* auxtrace section: start addr */
+ 0x00000018, 0x00000000, /* auxtrace section: len */
+ 0x000027dc, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+};
+
+static unsigned int perf_sections_feat[] = {
+ 0x00000000, 0x00640001,
+ 0xffffffff, 0xc9b5ee32,
+ 0xa6009dc9, 0xcb21093d,
+ 0x23c315d8, 0x1067c385,
+ 0x00000000, 0x72656b5b,
+ 0x2e6c656e, 0x6c6c616b,
+ 0x736d7973, 0x0000005d,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000, 0x00000000,
+ 0x00000000,
+
+ 0x00000001, 0x00000000,
+ 0x000006a8, 0x00000000,
+ 0x00000030, 0x00000000,
+};
+
+#define koffset(struct, member) struct##_##member##_offset
+
+/* at = ((struct *)ptr)->member */
+#define read_value(at, ptr, struct, member) \
+ do { \
+ readmem(ptr + koffset(struct, member), KVADDR, \
+ &at, sizeof(at), #struct "'s " #member, \
+ RETURN_ON_ERROR); \
+ } while (0)
+
+
+#define init_offset(struct, member) do { \
+ koffset(struct, member) = MEMBER_OFFSET(#struct, #member);\
+ if (koffset(struct, member) < 0) { \
+ fprintf(fp, "failed to init the offset, struct:"\
+ #struct ", member:" #member); \
+ fprintf(fp, "\n"); \
+ return -1; \
+ } \
+ } while (0)
+
+static int koffset(coresight_dump_node, cpu);
+static int koffset(coresight_dump_node, list);
+static int koffset(coresight_dump_node, buf);
+static int koffset(coresight_dump_node, buf_size);
+static int koffset(coresight_dump_node, name);
+static int koffset(coresight_dump_node, csdev);
+
+static int koffset(coresight_device, dev);
+static int koffset(device, kobj);
+static int koffset(kobject, name);
+
+static struct list_data list_data;
+static int instance_count;
+static int csdump_metadata_len = 0;
+
+static int csdump_write_buf(FILE *out_fp, char *component, int cpu_idx)
+{
+ ulong instance_ptr;
+ ulong field;
+ ulong buf_addr;
+ ulong csdev_addr;
+ ulong dev_addr;
+ ulong kobj_addr;
+ ulong name_addr;
+ int cpu, buf_sz, i, ret;
+ char name[64];
+ char *buf;
+
+ /* We start i at 1 to skip over the list_head and continue to the last
+ * instance, which lies at index instance_count */
+ for (i = 1; i <= instance_count; i++) {
+ instance_ptr = list_data.list_ptr[i];
+
+ field = instance_ptr - koffset(coresight_dump_node, list);
+
+ read_value(cpu, field, coresight_dump_node, cpu);
+ read_value(buf_addr, field, coresight_dump_node, buf);
+ read_value(buf_sz, field, coresight_dump_node, buf_size);
+
+ read_value(csdev_addr, field, coresight_dump_node, csdev);
+ dev_addr = csdev_addr + koffset(coresight_device, dev);
+ kobj_addr = dev_addr + koffset(device, kobj);
+
+ read_value(name_addr, kobj_addr, kobject, name);
+ read_string(name_addr, name, 64);
+
+ if (!buf_sz)
+ continue;
+
+ if (strstr(name, component) && (cpu == cpu_idx))
+ break;
+ }
+
+ if (i > instance_count)
+ return -1;
+
+ buf = malloc(buf_sz);
+ readmem(buf_addr, KVADDR, buf, buf_sz, "read dump log buf",
+ FAULT_ON_ERROR);
+
+ ret = fwrite(buf, buf_sz, 1, out_fp);
+ if (!ret) {
+ fprintf(fp, "[%d] Cannot write file\n", cpu);
+ free(buf);
+ return -1;
+ }
+
+ free(buf);
+
+ return buf_sz;
+}
+
+static int csdump_metadata(void)
+{
+ FILE *out_fp;
+ int online_cpus, i;
+
+ if ((out_fp = fopen("./metadata.bin", "w")) == NULL) {
+ fprintf(fp, "Cannot open file\n");
+ return -1;
+ }
+
+ online_cpus = get_cpus_online();
+ for (i = 0; i < online_cpus; i++) {
+ fprintf(fp, "cpu = %d\n", i);
+ csdump_metadata_len += csdump_write_buf(out_fp, "etm", i);
+ }
+
+ fclose(out_fp);
+
+ return 0;
+}
+
+static int csdump_tracedata(void)
+{
+ FILE *out_fp;
+
+ if ((out_fp = fopen("./cstrace.bin", "w")) == NULL) {
+ fprintf(fp, "Cannot open file\n");
+ return -1;
+ }
+
+ csdump_write_buf(out_fp, "etf", 0);
+
+ fclose(out_fp);
+
+ return 0;
+}
+
+static int csdump_perfdata(void)
+{
+ FILE *out_fp;
+ int online_cpus, i;
+ int trace_len = 0;
+ int pos, diff;
+
+ if ((out_fp = fopen("./perf.data", "w")) == NULL) {
+ fprintf(fp, "Cannot open file\n");
+ return -1;
+ }
+
+ fwrite(perf_header, sizeof(perf_header), 1, out_fp);
+ fwrite(perf_event_id, sizeof(perf_event_id), 1, out_fp);
+ fwrite(perf_event_cs_etm, sizeof(perf_event_cs_etm), 1, out_fp);
+ fwrite(perf_event_dummy, sizeof(perf_event_dummy), 1, out_fp);
+
+ online_cpus = get_cpus_online();
+
+ /* Adjust auxtrace_info size */
+ perf_auxtrace_info[1] = (perf_auxtrace_info[1] & 0xffff) |
+ ((sizeof(perf_auxtrace_info) + csdump_metadata_len) << 16);
+ /* Adjust CPU num */
+ perf_auxtrace_info[6] = online_cpus;
+
+ fwrite(perf_auxtrace_info, sizeof(perf_auxtrace_info), 1, out_fp);
+ trace_len += sizeof(perf_auxtrace_info);
+
+ for (i = 0; i < online_cpus; i++) {
+ fprintf(fp, "cpu = %d\n", i);
+ trace_len += csdump_write_buf(out_fp, "etm", i);
+ }
+
+ fwrite(perf_kernel_mmap, sizeof(perf_kernel_mmap), 1, out_fp);
+ trace_len += sizeof(perf_kernel_mmap);
+
+ fwrite(perf_threads, sizeof(perf_threads), 1, out_fp);
+ trace_len += sizeof(perf_threads);
+
+ fwrite(perf_auxtrace_snapshot, sizeof(perf_auxtrace_snapshot), 1, out_fp);
+ trace_len += sizeof(perf_auxtrace_snapshot);
+
+ trace_len += csdump_write_buf(out_fp, "etf", 0);
+
+ fwrite(perf_record_finish, sizeof(perf_record_finish), 1, out_fp);
+ trace_len += sizeof(perf_record_finish);
+
+ pos = ftell(out_fp);
+ pos += sizeof(perf_sections);
+
+ diff = perf_sections[0] - pos;
+
+ for (i = 0; i < sizeof(perf_sections) / 4; i += 4) {
+ if (!perf_sections[i])
+ continue;
+
+ perf_sections[i] = perf_sections[i] - diff;
+ }
+
+ fwrite(perf_sections, sizeof(perf_sections), 1, out_fp);
+ fwrite(perf_sections_feat, sizeof(perf_sections_feat), 1, out_fp);
+
+ fseek(out_fp, 48L, SEEK_SET);
+ fwrite(&trace_len, sizeof(trace_len), 1, out_fp);
+
+ fclose(out_fp);
+
+ return 0;
+}
+
+static int csdump_prepare(void)
+{
+ struct syment *sym_dump_list;
+ struct kernel_list_head *cs_dump_list_head;
+
+ init_offset(coresight_dump_node, cpu);
+ init_offset(coresight_dump_node, list);
+ init_offset(coresight_dump_node, buf);
+ init_offset(coresight_dump_node, buf_size);
+ init_offset(coresight_dump_node, name);
+ init_offset(coresight_dump_node, csdev);
+
+ init_offset(coresight_device, dev);
+ init_offset(device, kobj);
+ init_offset(kobject, name);
+
+ /* Get pointer to dump list */
+ sym_dump_list = symbol_search("coresight_dump_list");
+ if (!sym_dump_list) {
+ fprintf(fp, "symbol coresight_dump_list is not found\n");
+ return -1;
+ }
+
+ cs_dump_list_head = (void *)sym_dump_list->value;
+ fprintf(fp, "cs_dump_list_head = 0x%p\n", cs_dump_list_head);
+
+ BZERO(&list_data, sizeof(struct list_data));
+ list_data.start = (ulong)cs_dump_list_head;
+ list_data.end = (ulong)cs_dump_list_head;
+ list_data.flags = LIST_ALLOCATE;
+ instance_count = do_list(&list_data);
+
+ /*
+ * The do_list count includes the list_head, which is not
+ * a proper instance so minus 1.
+ */
+ instance_count--;
+ if (instance_count <= 0)
+ return -1;
+
+ return 0;
+}
+
+static void csdump_unprepare(void)
+{
+ FREEBUF(list_data.list_ptr);
+}
+
+void cmd_csdump(void)
+{
+ char* outdir;
+ mode_t mode = S_IRUSR | S_IWUSR | S_IXUSR |
+ S_IRGRP | S_IXGRP |
+ S_IROTH | S_IXOTH; /* 0755 */
+ int ret;
+
+ if (argcnt != 2)
+ cmd_usage(pc->curcmd, SYNOPSIS);
+
+ outdir = args[1];
+ if ((ret = mkdir(outdir, mode))) {
+ fprintf(fp, "Cannot create directory %s: %d\n", outdir, ret);
+ return;
+ }
+
+ if ((ret = chdir(outdir))) {
+ fprintf(fp, "Cannot chdir %s: %d\n", outdir, ret);
+ return;
+ }
+
+ if (csdump_prepare())
+ goto out;
+
+ csdump_metadata();
+ csdump_tracedata();
+ csdump_perfdata();
+
+out:
+ csdump_unprepare();
+ chdir("..");
+ return;
+}
+
+char *help_csdump[] = {
+ "csdump",
+ "Dump log buffer of Coresight Trace",
+ "<output-dir>",
+ "This command extracts coresight log buffer to the directory",
+ "specified by <output-dir>",
+ NULL
+};
+
+static struct command_table_entry command_table[] = {
+ { "csdump", cmd_csdump, help_csdump, 0},
+ { NULL },
+};
+
+void __attribute__((constructor))
+csdump_init(void)
+{
+ register_extension(command_table);
+}
+
+void __attribute__((destructor))
+csdump_fini(void) { }
--
2.7.4
7 years, 1 month
[PATCH v2 0/1] Display relative lag of each CPU
by Oleksandr Natalenko
While analyzing vmcore it is useful to have an information
about relative lag of each CPU. Usually, people do something like this:
runq -t | grep CPU | sort -k3r |
awk 'NR==1{now=strtonum("0x"$3)}1{printf"%s\t%7.2fs behind\n",
$0,(now-strtonum("0x"$3))/1000000000}'
Here, most recent runqueue timestamp is taken as a basis, and other
runqueues are assumed to lag behind it. This information can tell
the reviewer which CPU is experiencing some lockup, which is
especially useful for vmcores taken in virtual machines.
I think it would be nice to have this feature implemented by crash
utility itself.
Since this is an RFC and if this enhancement looks useful, I'd like
to ask to assist me with output indentation since dump_on_rq_timestamp(),
for instance, looks too implicit and (honestly) awful to use as it is.
Changes since v1:
* use calculated amount of leading spaces to indent output nicely
Oleksandr Natalenko (1):
runq: display relative lag of each CPU
help.c | 4 ++-
task.c | 93 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++-
2 files changed, 95 insertions(+), 2 deletions(-)
--
2.15.0
7 years, 1 month
Re: [Crash-utility] [PATCH v2] book3s/ppc64: update hash page table geometry
by anderson
Yep, I noticed that too late. I'll change it to 4.12 in the crash.changelog file when I release 7.2.1.
Sent from my Verizon, Samsung Galaxy smartphone
-------- Original message --------From: Hari Bathini <hbathini(a)linux.vnet.ibm.com> Date: 11/9/17 12:14 PM (GMT-05:00) To: Dave Anderson <anderson(a)redhat.com> Cc: crash-utility <crash-utility(a)redhat.com> Subject: Re: [Crash-utility] [PATCH v2] book3s/ppc64: update hash page table
geometry
On Thursday 09 November 2017 10:13 PM, Dave Anderson wrote:
>
> ----- Original Message -----
>> Starting with kernel 4.12, the hash page table geometry is updated to
>> accommodate larger VA range. Update here accordingly.
>>
>> Signed-off-by: Hari Bathini <hbathini(a)linux.vnet.ibm.com>
>> ---
>> changes in v1:
>> * Kernel commit 92d9dfda8b54 reverted the geometry update
>> for 4K hash pagetable. So, undo'ing 4K pagesize changes.
> Hari,
>
> As discussed in the related bugzilla, this patch fixes the user-space address
> translation issues seen in 4.14-based kernels. Queued for crash-7.2.1:
>
> https://github.com/crash-utility/crash/commit/c8178eca9c74f81a7f803a58d33...
Hi Dave,
I guess, the changelog should read *4.12 and later* instead of *4.14 and
later* ...
Thanks for taking the patch.
- Hari
--
Crash-utility mailing list
Crash-utility(a)redhat.com
https://www.redhat.com/mailman/listinfo/crash-utility
7 years, 1 month
[PATCH v2] book3s/ppc64: update hash page table geometry
by Hari Bathini
Starting with kernel 4.12, the hash page table geometry is updated to
accommodate larger VA range. Update here accordingly.
Signed-off-by: Hari Bathini <hbathini(a)linux.vnet.ibm.com>
---
changes in v1:
* Kernel commit 92d9dfda8b54 reverted the geometry update
for 4K hash pagetable. So, undo'ing 4K pagesize changes.
defs.h | 3 +++
ppc64.c | 12 +++++++++---
2 files changed, 12 insertions(+), 3 deletions(-)
diff --git a/defs.h b/defs.h
index 18f36b3..9132075 100644
--- a/defs.h
+++ b/defs.h
@@ -3915,6 +3915,9 @@ struct efi_memory_desc_t {
#define PGD_INDEX_SIZE_L4_64K_3_10 12
#define PMD_INDEX_SIZE_L4_64K_4_6 5
#define PUD_INDEX_SIZE_L4_64K_4_6 5
+#define PMD_INDEX_SIZE_L4_64K_4_12 10
+#define PUD_INDEX_SIZE_L4_64K_4_12 7
+#define PGD_INDEX_SIZE_L4_64K_4_12 8
#define PTE_INDEX_SIZE_RADIX_64K 5
#define PMD_INDEX_SIZE_RADIX_64K 9
#define PUD_INDEX_SIZE_RADIX_64K 9
diff --git a/ppc64.c b/ppc64.c
index 84cec09..672ee60 100644
--- a/ppc64.c
+++ b/ppc64.c
@@ -447,10 +447,16 @@ ppc64_init(int when)
} else if (!(machdep->flags & BOOK3E) &&
(THIS_KERNEL_VERSION >= LINUX(4,6,0))) {
m->l1_index_size = PTE_INDEX_SIZE_L4_64K_3_10;
- m->l2_index_size = PMD_INDEX_SIZE_L4_64K_4_6;
- m->l3_index_size = PUD_INDEX_SIZE_L4_64K_4_6;
- m->l4_index_size = PGD_INDEX_SIZE_L4_64K_3_10;
+ if (THIS_KERNEL_VERSION >= LINUX(4,12,0)) {
+ m->l2_index_size = PMD_INDEX_SIZE_L4_64K_4_12;
+ m->l3_index_size = PUD_INDEX_SIZE_L4_64K_4_12;
+ m->l4_index_size = PGD_INDEX_SIZE_L4_64K_4_12;
+ } else {
+ m->l2_index_size = PMD_INDEX_SIZE_L4_64K_4_6;
+ m->l3_index_size = PUD_INDEX_SIZE_L4_64K_4_6;
+ m->l4_index_size = PGD_INDEX_SIZE_L4_64K_3_10;
+ }
} else if (THIS_KERNEL_VERSION >= LINUX(3,10,0)) {
m->l1_index_size = PTE_INDEX_SIZE_L4_64K_3_10;
m->l2_index_size = PMD_INDEX_SIZE_L4_64K_3_10;
7 years, 1 month
[PATCH RFC 0/1] Display relative lag of each CPU
by Oleksandr Natalenko
While analyzing vmcore it is useful to have an information
about relative lag of each CPU. Usually, people do something like this:
runq -t | grep CPU | sort -k3r |
awk 'NR==1{now=strtonum("0x"$3)}1{printf"%s\t%7.2fs behind\n",
$0,(now-strtonum("0x"$3))/1000000000}'
Here, most recent runqueue timestamp is taken as a basis, and other
runqueues are assumed to lag behind it. This information can tell
the reviewer which CPU is experiencing some lockup, which is
especially useful for vmcores taken in virtual machines.
I think it would be nice to have this feature implemented by crash
utility itself.
Since this is an RFC and if this enhancement looks useful, I'd like
to ask to assist me with output indentation since dump_on_rq_timestamp(),
for instance, looks too implicit and (honestly) awful to use as it is.
Oleksandr Natalenko (1):
runq: display relative lag of each CPU
help.c | 4 +++-
task.c | 76 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++-
2 files changed, 78 insertions(+), 2 deletions(-)
--
2.15.0
7 years, 1 month
[PATCH] book3s/ppc64: update hash page table geometry
by Hari Bathini
Starting with kernel 4.12, the hash page table geometry is updated to
accommodate larger VA range. Update here accordingly.
Signed-off-by: Hari Bathini <hbathini(a)linux.vnet.ibm.com>
---
defs.h | 4 ++++
ppc64.c | 18 ++++++++++++++----
2 files changed, 18 insertions(+), 4 deletions(-)
diff --git a/defs.h b/defs.h
index 18f36b3..0bf718c 100644
--- a/defs.h
+++ b/defs.h
@@ -3895,6 +3895,7 @@ struct efi_memory_desc_t {
#define PUD_INDEX_SIZE_L4_4K 7
#define PGD_INDEX_SIZE_L4_4K 9
#define PUD_INDEX_SIZE_L4_4K_3_7 9
+#define PGD_INDEX_SIZE_L4_4K_4_12 12
#define PTE_INDEX_SIZE_RADIX_4K 9
#define PMD_INDEX_SIZE_RADIX_4K 9
#define PUD_INDEX_SIZE_RADIX_4K 9
@@ -3915,6 +3916,9 @@ struct efi_memory_desc_t {
#define PGD_INDEX_SIZE_L4_64K_3_10 12
#define PMD_INDEX_SIZE_L4_64K_4_6 5
#define PUD_INDEX_SIZE_L4_64K_4_6 5
+#define PMD_INDEX_SIZE_L4_64K_4_12 10
+#define PUD_INDEX_SIZE_L4_64K_4_12 7
+#define PGD_INDEX_SIZE_L4_64K_4_12 8
#define PTE_INDEX_SIZE_RADIX_64K 5
#define PMD_INDEX_SIZE_RADIX_64K 9
#define PUD_INDEX_SIZE_RADIX_64K 9
diff --git a/ppc64.c b/ppc64.c
index 84cec09..727749e 100644
--- a/ppc64.c
+++ b/ppc64.c
@@ -447,10 +447,16 @@ ppc64_init(int when)
} else if (!(machdep->flags & BOOK3E) &&
(THIS_KERNEL_VERSION >= LINUX(4,6,0))) {
m->l1_index_size = PTE_INDEX_SIZE_L4_64K_3_10;
- m->l2_index_size = PMD_INDEX_SIZE_L4_64K_4_6;
- m->l3_index_size = PUD_INDEX_SIZE_L4_64K_4_6;
- m->l4_index_size = PGD_INDEX_SIZE_L4_64K_3_10;
+ if (THIS_KERNEL_VERSION >= LINUX(4,12,0)) {
+ m->l2_index_size = PMD_INDEX_SIZE_L4_64K_4_12;
+ m->l3_index_size = PUD_INDEX_SIZE_L4_64K_4_12;
+ m->l4_index_size = PGD_INDEX_SIZE_L4_64K_4_12;
+ } else {
+ m->l2_index_size = PMD_INDEX_SIZE_L4_64K_4_6;
+ m->l3_index_size = PUD_INDEX_SIZE_L4_64K_4_6;
+ m->l4_index_size = PGD_INDEX_SIZE_L4_64K_3_10;
+ }
} else if (THIS_KERNEL_VERSION >= LINUX(3,10,0)) {
m->l1_index_size = PTE_INDEX_SIZE_L4_64K_3_10;
m->l2_index_size = PMD_INDEX_SIZE_L4_64K_3_10;
@@ -499,7 +505,11 @@ ppc64_init(int when)
m->l3_index_size = PUD_INDEX_SIZE_L4_4K_3_7;
else
m->l3_index_size = PUD_INDEX_SIZE_L4_4K;
- m->l4_index_size = PGD_INDEX_SIZE_L4_4K;
+ if (!(machdep->flags & BOOK3E) &&
+ (THIS_KERNEL_VERSION >= LINUX(4,12,0)))
+ m->l4_index_size = PGD_INDEX_SIZE_L4_4K_4_12;
+ else
+ m->l4_index_size = PGD_INDEX_SIZE_L4_4K;
if (machdep->flags & BOOK3E)
m->pte_rpn_shift = PTE_RPN_SHIFT_L4_BOOK3E_4K;
7 years, 1 month
[PATCH v2 0/1] Add option to filter disks with no I/O in progress
by Oleksandr Natalenko
While analyzing vmcores the first thing that people do when they want
to check disk I/O is something like this:
crash> dev -d | awk '$5 != 0'
Since there might be lots of disks, this makes scrolling through them
much easier.
I think we should have this option implemented for dev command itself.
Changes since v1:
* add new flag to the list of all flags in help output
* add missing dot in the new flag description
Oleksandr Natalenko (1):
dev: add option to filter disks with no I/O
dev.c | 27 +++++++++++++++++++--------
help.c | 3 ++-
2 files changed, 21 insertions(+), 9 deletions(-)
--
2.15.0
7 years, 1 month