lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [thread-next>] [day] [month] [year] [list]
Message-ID: <20111005182503.GA6142@google.com>
Date:	Wed, 5 Oct 2011 14:25:03 -0400
From:	Thomas Tuttle <ttuttle@...omium.org>
To:	linux-kernel@...r.kernel.org
Cc:	Marco Stornelli <marco.stornelli@...il.com>
Subject: [PATCH] ramoops: scripts/ramoops.c for extracting oopses

Add a C program to extract oopses stored by ramoops.  Running with no
arguments will list all of the records; running with an argument number
will retrieve the oops from that record, if it is valid.

(In the long term, it'd be ideal to have some kernel interface to
ramoops records, but this script may be useful in the short term.)

BUG=chromium-os:21113
TEST=Adhoc, seems to work :)

Change-Id: I24edf4d09c96a5d89bb75859a83d5427502d4eb6
Signed-off-by: Thomas Tuttle <ttuttle@...omium.org>
---
 scripts/ramoops.c |  175 +++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 175 insertions(+), 0 deletions(-)
 create mode 100644 scripts/ramoops.c

diff --git a/scripts/ramoops.c b/scripts/ramoops.c
new file mode 100644
index 0000000..25cc7c5
--- /dev/null
+++ b/scripts/ramoops.c
@@ -0,0 +1,175 @@
+/*
+ * ramoops: Extract ramoops images from /dev/mem.
+ *
+ * Copyright (C) 2011 Thomas Tuttle <ttuttle@...gle.com>
+ *
+ * This file is subject to the terms and conditions of the GNU General Public
+ * License. See the file COPYING in the main directory of the Linux
+ * distribution for more details.
+ */
+
+#include <errno.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+static char *progn;
+
+static void read_long(char *name, unsigned long *out)
+{
+	FILE *f = fopen(name, "r");
+	if (!f) {
+		fprintf(stderr, "%s: %s: %s\n",
+			progn, name, strerror(errno));
+		exit(EXIT_FAILURE);
+	}
+
+	if (fscanf(f, "%lu", out) < 1) {
+		fprintf(stderr, "%s: %s: couldn't read a number\n",
+			progn, name);
+		exit(EXIT_FAILURE);
+	}
+
+	fclose(f);
+}
+
+static const char *MEM_NAME = "/dev/mem";
+
+static FILE *open_mem(void)
+{
+	FILE *mem = fopen(MEM_NAME, "r");
+	if (!mem) {
+		fprintf(stderr, "%s: open %s: %s\n",
+			progn, MEM_NAME, strerror(errno));
+		exit(EXIT_FAILURE);
+	}
+	return mem;
+}
+
+static void seek_mem(FILE *mem, unsigned long off)
+{
+	if (fseek(mem, off, SEEK_SET) < 0) {
+		fprintf(stderr, "%s: seek %s: %s\n",
+			progn, MEM_NAME, strerror(errno));
+		exit(EXIT_FAILURE);
+	}
+}
+
+static void read_mem(FILE *mem, char *buf, unsigned long len)
+{
+	size_t res;
+
+	res = fread(buf, 1, len, mem);
+	if (res == 0 && ferror(mem)) {
+		fprintf(stderr, "%s: read %s: %s\n",
+			progn, MEM_NAME, strerror(errno));
+		exit(EXIT_FAILURE);
+	} else if (res < len) {
+		fprintf(stderr, "%s: read %s: short read\n",
+			progn, MEM_NAME);
+		exit(EXIT_FAILURE);
+	}
+}
+
+static void putsn(const char *s, size_t n)
+{
+	while (*s && n-- > 0)
+		putchar(*s++);
+}
+
+static int check_record(char *buf, char *time_out, size_t time_len)
+{
+	static const char *MAGIC = "====";
+	static const size_t MAGIC_LEN = 4;
+
+	char *s;
+
+	if (memcmp(buf, MAGIC, MAGIC_LEN))
+		return 0;
+
+	s = buf + MAGIC_LEN;
+	while (*s != '\n' && --time_len > 0)
+		*time_out++ = *s++;
+	*time_out = '\0';
+
+	return 1;
+}
+
+static const int REC_LIST = -1;
+
+#define RAMOOPS_PARAM "/sys/module/ramoops/parameters/"
+
+static int print_record(int record, int contents, FILE *mem,
+			unsigned long record_size, char *buf)
+{
+	static const size_t TIME_LEN = 64;
+
+	char time[TIME_LEN];
+
+	read_mem(mem, buf, record_size);
+	fprintf(stderr, "ramoops record %d: ", record);
+	if (check_record(buf, time, TIME_LEN)) {
+		fprintf(stderr, "valid, timestamp %s\n", time);
+		if (contents)
+			putsn(buf, record_size);
+		return 0;
+	} else {
+		fprintf(stderr, "invalid\n");
+		return 1;
+	}
+}
+
+static int print_records(int record)
+{
+	unsigned long mem_address, mem_size, record_size, num_records;
+	FILE *mem;
+	int i;
+	int ret = 0;
+
+	read_long(RAMOOPS_PARAM "mem_address", &mem_address);
+	read_long(RAMOOPS_PARAM "mem_size",    &mem_size);
+	read_long(RAMOOPS_PARAM "record_size", &record_size);
+	num_records = mem_size / record_size;
+
+	if (record != REC_LIST &&
+	    (record < 0 || (unsigned long)record >= num_records)) {
+		fprintf(stderr, "%s: record number %d out of range [0, %ld)\n",
+			progn, record, num_records);
+		return 1;
+	}
+
+	buf = malloc(record_size);
+	if (!buf) {
+		fprintf(stderr, "%s: failed to allocate record buffer\n",
+			progn);
+		return 1;
+	}
+
+	mem = open_mem();
+	if (record == REC_LIST) {
+		seek_mem(mem, mem_address);
+		for (i = 0; i < (int)num_records; i++)
+			print_record(i, 0, mem, record_size, buf);
+	} else {
+		seek_mem(mem, mem_address + record * record_size);
+		ret = print_record(record, 1, mem, record_size, buf);
+	}
+	fclose(mem);
+
+	return ret;
+}
+
+int main(int argc, char *argv[])
+{
+	progn = argv[0];
+
+	switch (argc) {
+	case 1:
+		return print_records(REC_LIST);
+	case 2:
+		return print_records(atoi(argv[1]));
+	default:
+		fprintf(stderr, "Usage: %s [record]\n", progn);
+		return 1;
+	}
+}
-- 
1.7.3.1

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ