607 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			607 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* drivers/misc/apanic.c
 | 
						|
 *
 | 
						|
 * Copyright (C) 2009 Google, Inc.
 | 
						|
 * Author: San Mehat <san@android.com>
 | 
						|
 *
 | 
						|
 * This software is licensed under the terms of the GNU General Public
 | 
						|
 * License version 2, as published by the Free Software Foundation, and
 | 
						|
 * may be copied, distributed, and modified under those terms.
 | 
						|
 *
 | 
						|
 * 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.
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/module.h>
 | 
						|
#include <linux/kernel.h>
 | 
						|
#include <linux/string.h>
 | 
						|
#include <linux/errno.h>
 | 
						|
#include <linux/init.h>
 | 
						|
#include <linux/interrupt.h>
 | 
						|
#include <linux/device.h>
 | 
						|
#include <linux/types.h>
 | 
						|
#include <linux/delay.h>
 | 
						|
#include <linux/sched.h>
 | 
						|
#include <linux/wait.h>
 | 
						|
#include <linux/wakelock.h>
 | 
						|
#include <linux/platform_device.h>
 | 
						|
#include <linux/uaccess.h>
 | 
						|
#include <linux/mtd/mtd.h>
 | 
						|
#include <linux/notifier.h>
 | 
						|
#include <linux/mtd/mtd.h>
 | 
						|
#include <linux/debugfs.h>
 | 
						|
#include <linux/fs.h>
 | 
						|
#include <linux/proc_fs.h>
 | 
						|
#include <linux/mutex.h>
 | 
						|
#include <linux/workqueue.h>
 | 
						|
#include <linux/preempt.h>
 | 
						|
 | 
						|
extern void ram_console_enable_console(int);
 | 
						|
 | 
						|
struct panic_header {
 | 
						|
	u32 magic;
 | 
						|
#define PANIC_MAGIC 0xdeadf00d
 | 
						|
 | 
						|
	u32 version;
 | 
						|
#define PHDR_VERSION   0x01
 | 
						|
 | 
						|
	u32 console_offset;
 | 
						|
	u32 console_length;
 | 
						|
 | 
						|
	u32 threads_offset;
 | 
						|
	u32 threads_length;
 | 
						|
};
 | 
						|
 | 
						|
struct apanic_data {
 | 
						|
	struct mtd_info		*mtd;
 | 
						|
	struct panic_header	curr;
 | 
						|
	void			*bounce;
 | 
						|
	struct proc_dir_entry	*apanic_console;
 | 
						|
	struct proc_dir_entry	*apanic_threads;
 | 
						|
};
 | 
						|
 | 
						|
static struct apanic_data drv_ctx;
 | 
						|
static struct work_struct proc_removal_work;
 | 
						|
static DEFINE_MUTEX(drv_mutex);
 | 
						|
 | 
						|
static unsigned int *apanic_bbt;
 | 
						|
static unsigned int apanic_erase_blocks;
 | 
						|
static unsigned int apanic_good_blocks;
 | 
						|
 | 
						|
static void set_bb(unsigned int block, unsigned int *bbt)
 | 
						|
{
 | 
						|
	unsigned int flag = 1;
 | 
						|
 | 
						|
	BUG_ON(block >= apanic_erase_blocks);
 | 
						|
 | 
						|
	flag = flag << (block%32);
 | 
						|
	apanic_bbt[block/32] |= flag;
 | 
						|
	apanic_good_blocks--;
 | 
						|
}
 | 
						|
 | 
						|
static unsigned int get_bb(unsigned int block, unsigned int *bbt)
 | 
						|
{
 | 
						|
	unsigned int flag;
 | 
						|
 | 
						|
	BUG_ON(block >= apanic_erase_blocks);
 | 
						|
 | 
						|
	flag = 1 << (block%32);
 | 
						|
	return apanic_bbt[block/32] & flag;
 | 
						|
}
 | 
						|
 | 
						|
static void alloc_bbt(struct mtd_info *mtd, unsigned int *bbt)
 | 
						|
{
 | 
						|
	int bbt_size;
 | 
						|
	apanic_erase_blocks = (mtd->size)>>(mtd->erasesize_shift);
 | 
						|
	bbt_size = (apanic_erase_blocks+32)/32;
 | 
						|
 | 
						|
	apanic_bbt = kmalloc(bbt_size*4, GFP_KERNEL);
 | 
						|
	memset(apanic_bbt, 0, bbt_size*4);
 | 
						|
	apanic_good_blocks = apanic_erase_blocks;
 | 
						|
}
 | 
						|
static void scan_bbt(struct mtd_info *mtd, unsigned int *bbt)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < apanic_erase_blocks; i++) {
 | 
						|
		if (mtd->block_isbad(mtd, i*mtd->erasesize))
 | 
						|
			set_bb(i, apanic_bbt);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
#define APANIC_INVALID_OFFSET 0xFFFFFFFF
 | 
						|
 | 
						|
static unsigned int phy_offset(struct mtd_info *mtd, unsigned int offset)
 | 
						|
{
 | 
						|
	unsigned int logic_block = offset>>(mtd->erasesize_shift);
 | 
						|
	unsigned int phy_block;
 | 
						|
	unsigned good_block = 0;
 | 
						|
 | 
						|
	for (phy_block = 0; phy_block < apanic_erase_blocks; phy_block++) {
 | 
						|
		if (!get_bb(phy_block, apanic_bbt))
 | 
						|
			good_block++;
 | 
						|
		if (good_block == (logic_block + 1))
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	if (good_block != (logic_block + 1))
 | 
						|
		return APANIC_INVALID_OFFSET;
 | 
						|
 | 
						|
	return offset + ((phy_block-logic_block)<<mtd->erasesize_shift);
 | 
						|
}
 | 
						|
 | 
						|
static void apanic_erase_callback(struct erase_info *done)
 | 
						|
{
 | 
						|
	wait_queue_head_t *wait_q = (wait_queue_head_t *) done->priv;
 | 
						|
	wake_up(wait_q);
 | 
						|
}
 | 
						|
 | 
						|
static int apanic_proc_read(char *buffer, char **start, off_t offset,
 | 
						|
			       int count, int *peof, void *dat)
 | 
						|
{
 | 
						|
	struct apanic_data *ctx = &drv_ctx;
 | 
						|
	size_t file_length;
 | 
						|
	off_t file_offset;
 | 
						|
	unsigned int page_no;
 | 
						|
	off_t page_offset;
 | 
						|
	int rc;
 | 
						|
	size_t len;
 | 
						|
 | 
						|
	if (!count)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	mutex_lock(&drv_mutex);
 | 
						|
 | 
						|
	switch ((int) dat) {
 | 
						|
	case 1:	/* apanic_console */
 | 
						|
		file_length = ctx->curr.console_length;
 | 
						|
		file_offset = ctx->curr.console_offset;
 | 
						|
		break;
 | 
						|
	case 2:	/* apanic_threads */
 | 
						|
		file_length = ctx->curr.threads_length;
 | 
						|
		file_offset = ctx->curr.threads_offset;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		pr_err("Bad dat (%d)\n", (int) dat);
 | 
						|
		mutex_unlock(&drv_mutex);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	if ((offset + count) > file_length) {
 | 
						|
		mutex_unlock(&drv_mutex);
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	/* We only support reading a maximum of a flash page */
 | 
						|
	if (count > ctx->mtd->writesize)
 | 
						|
		count = ctx->mtd->writesize;
 | 
						|
 | 
						|
	page_no = (file_offset + offset) / ctx->mtd->writesize;
 | 
						|
	page_offset = (file_offset + offset) % ctx->mtd->writesize;
 | 
						|
 | 
						|
 | 
						|
	if (phy_offset(ctx->mtd, (page_no * ctx->mtd->writesize))
 | 
						|
		== APANIC_INVALID_OFFSET) {
 | 
						|
		pr_err("apanic: reading an invalid address\n");
 | 
						|
		mutex_unlock(&drv_mutex);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	rc = ctx->mtd->read(ctx->mtd,
 | 
						|
		phy_offset(ctx->mtd, (page_no * ctx->mtd->writesize)),
 | 
						|
		ctx->mtd->writesize,
 | 
						|
		&len, ctx->bounce);
 | 
						|
 | 
						|
	if (page_offset)
 | 
						|
		count -= page_offset;
 | 
						|
	memcpy(buffer, ctx->bounce + page_offset, count);
 | 
						|
 | 
						|
	*start = count;
 | 
						|
 | 
						|
	if ((offset + count) == file_length)
 | 
						|
		*peof = 1;
 | 
						|
 | 
						|
	mutex_unlock(&drv_mutex);
 | 
						|
	return count;
 | 
						|
}
 | 
						|
 | 
						|
static void mtd_panic_erase(void)
 | 
						|
{
 | 
						|
	struct apanic_data *ctx = &drv_ctx;
 | 
						|
	struct erase_info erase;
 | 
						|
	DECLARE_WAITQUEUE(wait, current);
 | 
						|
	wait_queue_head_t wait_q;
 | 
						|
	int rc, i;
 | 
						|
 | 
						|
	init_waitqueue_head(&wait_q);
 | 
						|
	erase.mtd = ctx->mtd;
 | 
						|
	erase.callback = apanic_erase_callback;
 | 
						|
	erase.len = ctx->mtd->erasesize;
 | 
						|
	erase.priv = (u_long)&wait_q;
 | 
						|
	for (i = 0; i < ctx->mtd->size; i += ctx->mtd->erasesize) {
 | 
						|
		erase.addr = i;
 | 
						|
		set_current_state(TASK_INTERRUPTIBLE);
 | 
						|
		add_wait_queue(&wait_q, &wait);
 | 
						|
 | 
						|
		if (get_bb(erase.addr>>ctx->mtd->erasesize_shift, apanic_bbt)) {
 | 
						|
			printk(KERN_WARNING
 | 
						|
			       "apanic: Skipping erase of bad "
 | 
						|
			       "block @%llx\n", erase.addr);
 | 
						|
			set_current_state(TASK_RUNNING);
 | 
						|
			remove_wait_queue(&wait_q, &wait);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		rc = ctx->mtd->erase(ctx->mtd, &erase);
 | 
						|
		if (rc) {
 | 
						|
			set_current_state(TASK_RUNNING);
 | 
						|
			remove_wait_queue(&wait_q, &wait);
 | 
						|
			printk(KERN_ERR
 | 
						|
			       "apanic: Erase of 0x%llx, 0x%llx failed\n",
 | 
						|
			       (unsigned long long) erase.addr,
 | 
						|
			       (unsigned long long) erase.len);
 | 
						|
			if (rc == -EIO) {
 | 
						|
				if (ctx->mtd->block_markbad(ctx->mtd,
 | 
						|
							    erase.addr)) {
 | 
						|
					printk(KERN_ERR
 | 
						|
					       "apanic: Err marking blk bad\n");
 | 
						|
					goto out;
 | 
						|
				}
 | 
						|
				printk(KERN_INFO
 | 
						|
				       "apanic: Marked a bad block"
 | 
						|
				       " @%llx\n", erase.addr);
 | 
						|
				set_bb(erase.addr>>ctx->mtd->erasesize_shift,
 | 
						|
					apanic_bbt);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			goto out;
 | 
						|
		}
 | 
						|
		schedule();
 | 
						|
		remove_wait_queue(&wait_q, &wait);
 | 
						|
	}
 | 
						|
	printk(KERN_DEBUG "apanic: %s partition erased\n",
 | 
						|
	       CONFIG_APANIC_PLABEL);
 | 
						|
out:
 | 
						|
	return;
 | 
						|
}
 | 
						|
 | 
						|
static void apanic_remove_proc_work(struct work_struct *work)
 | 
						|
{
 | 
						|
	struct apanic_data *ctx = &drv_ctx;
 | 
						|
 | 
						|
	mutex_lock(&drv_mutex);
 | 
						|
	mtd_panic_erase();
 | 
						|
	memset(&ctx->curr, 0, sizeof(struct panic_header));
 | 
						|
	if (ctx->apanic_console) {
 | 
						|
		remove_proc_entry("apanic_console", NULL);
 | 
						|
		ctx->apanic_console = NULL;
 | 
						|
	}
 | 
						|
	if (ctx->apanic_threads) {
 | 
						|
		remove_proc_entry("apanic_threads", NULL);
 | 
						|
		ctx->apanic_threads = NULL;
 | 
						|
	}
 | 
						|
	mutex_unlock(&drv_mutex);
 | 
						|
}
 | 
						|
 | 
						|
static int apanic_proc_write(struct file *file, const char __user *buffer,
 | 
						|
				unsigned long count, void *data)
 | 
						|
{
 | 
						|
	schedule_work(&proc_removal_work);
 | 
						|
	return count;
 | 
						|
}
 | 
						|
 | 
						|
static void mtd_panic_notify_add(struct mtd_info *mtd)
 | 
						|
{
 | 
						|
	struct apanic_data *ctx = &drv_ctx;
 | 
						|
	struct panic_header *hdr = ctx->bounce;
 | 
						|
	size_t len;
 | 
						|
	int rc;
 | 
						|
	int    proc_entry_created = 0;
 | 
						|
 | 
						|
	if (strcmp(mtd->name, CONFIG_APANIC_PLABEL))
 | 
						|
		return;
 | 
						|
 | 
						|
	ctx->mtd = mtd;
 | 
						|
 | 
						|
	alloc_bbt(mtd, apanic_bbt);
 | 
						|
	scan_bbt(mtd, apanic_bbt);
 | 
						|
 | 
						|
	if (apanic_good_blocks == 0) {
 | 
						|
		printk(KERN_ERR "apanic: no any good blocks?!\n");
 | 
						|
		goto out_err;
 | 
						|
	}
 | 
						|
 | 
						|
	rc = mtd->read(mtd, phy_offset(mtd, 0), mtd->writesize,
 | 
						|
			&len, ctx->bounce);
 | 
						|
	if (rc && rc == -EBADMSG) {
 | 
						|
		printk(KERN_WARNING
 | 
						|
		       "apanic: Bad ECC on block 0 (ignored)\n");
 | 
						|
	} else if (rc && rc != -EUCLEAN) {
 | 
						|
		printk(KERN_ERR "apanic: Error reading block 0 (%d)\n", rc);
 | 
						|
		goto out_err;
 | 
						|
	}
 | 
						|
 | 
						|
	if (len != mtd->writesize) {
 | 
						|
		printk(KERN_ERR "apanic: Bad read size (%d)\n", rc);
 | 
						|
		goto out_err;
 | 
						|
	}
 | 
						|
 | 
						|
	printk(KERN_INFO "apanic: Bound to mtd partition '%s'\n", mtd->name);
 | 
						|
 | 
						|
	if (hdr->magic != PANIC_MAGIC) {
 | 
						|
		printk(KERN_INFO "apanic: No panic data available\n");
 | 
						|
		mtd_panic_erase();
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	if (hdr->version != PHDR_VERSION) {
 | 
						|
		printk(KERN_INFO "apanic: Version mismatch (%d != %d)\n",
 | 
						|
		       hdr->version, PHDR_VERSION);
 | 
						|
		mtd_panic_erase();
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	memcpy(&ctx->curr, hdr, sizeof(struct panic_header));
 | 
						|
 | 
						|
	printk(KERN_INFO "apanic: c(%u, %u) t(%u, %u)\n",
 | 
						|
	       hdr->console_offset, hdr->console_length,
 | 
						|
	       hdr->threads_offset, hdr->threads_length);
 | 
						|
 | 
						|
	if (hdr->console_length) {
 | 
						|
		ctx->apanic_console = create_proc_entry("apanic_console",
 | 
						|
						      S_IFREG | S_IRUGO, NULL);
 | 
						|
		if (!ctx->apanic_console)
 | 
						|
			printk(KERN_ERR "%s: failed creating procfile\n",
 | 
						|
			       __func__);
 | 
						|
		else {
 | 
						|
			ctx->apanic_console->read_proc = apanic_proc_read;
 | 
						|
			ctx->apanic_console->write_proc = apanic_proc_write;
 | 
						|
			ctx->apanic_console->size = hdr->console_length;
 | 
						|
			ctx->apanic_console->data = (void *) 1;
 | 
						|
			proc_entry_created = 1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (hdr->threads_length) {
 | 
						|
		ctx->apanic_threads = create_proc_entry("apanic_threads",
 | 
						|
						       S_IFREG | S_IRUGO, NULL);
 | 
						|
		if (!ctx->apanic_threads)
 | 
						|
			printk(KERN_ERR "%s: failed creating procfile\n",
 | 
						|
			       __func__);
 | 
						|
		else {
 | 
						|
			ctx->apanic_threads->read_proc = apanic_proc_read;
 | 
						|
			ctx->apanic_threads->write_proc = apanic_proc_write;
 | 
						|
			ctx->apanic_threads->size = hdr->threads_length;
 | 
						|
			ctx->apanic_threads->data = (void *) 2;
 | 
						|
			proc_entry_created = 1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (!proc_entry_created)
 | 
						|
		mtd_panic_erase();
 | 
						|
 | 
						|
	return;
 | 
						|
out_err:
 | 
						|
	ctx->mtd = NULL;
 | 
						|
}
 | 
						|
 | 
						|
static void mtd_panic_notify_remove(struct mtd_info *mtd)
 | 
						|
{
 | 
						|
	struct apanic_data *ctx = &drv_ctx;
 | 
						|
	if (mtd == ctx->mtd) {
 | 
						|
		ctx->mtd = NULL;
 | 
						|
		printk(KERN_INFO "apanic: Unbound from %s\n", mtd->name);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static struct mtd_notifier mtd_panic_notifier = {
 | 
						|
	.add	= mtd_panic_notify_add,
 | 
						|
	.remove	= mtd_panic_notify_remove,
 | 
						|
};
 | 
						|
 | 
						|
static int in_panic = 0;
 | 
						|
 | 
						|
static int apanic_writeflashpage(struct mtd_info *mtd, loff_t to,
 | 
						|
				 const u_char *buf)
 | 
						|
{
 | 
						|
	int rc;
 | 
						|
	size_t wlen;
 | 
						|
	int panic = in_interrupt() | in_atomic();
 | 
						|
 | 
						|
	if (panic && !mtd->panic_write) {
 | 
						|
		printk(KERN_EMERG "%s: No panic_write available\n", __func__);
 | 
						|
		return 0;
 | 
						|
	} else if (!panic && !mtd->write) {
 | 
						|
		printk(KERN_EMERG "%s: No write available\n", __func__);
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	to = phy_offset(mtd, to);
 | 
						|
	if (to == APANIC_INVALID_OFFSET) {
 | 
						|
		printk(KERN_EMERG "apanic: write to invalid address\n");
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	if (panic)
 | 
						|
		rc = mtd->panic_write(mtd, to, mtd->writesize, &wlen, buf);
 | 
						|
	else
 | 
						|
		rc = mtd->write(mtd, to, mtd->writesize, &wlen, buf);
 | 
						|
 | 
						|
	if (rc) {
 | 
						|
		printk(KERN_EMERG
 | 
						|
		       "%s: Error writing data to flash (%d)\n",
 | 
						|
		       __func__, rc);
 | 
						|
		return rc;
 | 
						|
	}
 | 
						|
 | 
						|
	return wlen;
 | 
						|
}
 | 
						|
 | 
						|
extern int log_buf_copy(char *dest, int idx, int len);
 | 
						|
extern void log_buf_clear(void);
 | 
						|
 | 
						|
/*
 | 
						|
 * Writes the contents of the console to the specified offset in flash.
 | 
						|
 * Returns number of bytes written
 | 
						|
 */
 | 
						|
static int apanic_write_console(struct mtd_info *mtd, unsigned int off)
 | 
						|
{
 | 
						|
	struct apanic_data *ctx = &drv_ctx;
 | 
						|
	int saved_oip;
 | 
						|
	int idx = 0;
 | 
						|
	int rc, rc2;
 | 
						|
	unsigned int last_chunk = 0;
 | 
						|
 | 
						|
	while (!last_chunk) {
 | 
						|
		saved_oip = oops_in_progress;
 | 
						|
		oops_in_progress = 1;
 | 
						|
		rc = log_buf_copy(ctx->bounce, idx, mtd->writesize);
 | 
						|
		if (rc < 0)
 | 
						|
			break;
 | 
						|
 | 
						|
		if (rc != mtd->writesize)
 | 
						|
			last_chunk = rc;
 | 
						|
 | 
						|
		oops_in_progress = saved_oip;
 | 
						|
		if (rc <= 0)
 | 
						|
			break;
 | 
						|
		if (rc != mtd->writesize)
 | 
						|
			memset(ctx->bounce + rc, 0, mtd->writesize - rc);
 | 
						|
 | 
						|
		rc2 = apanic_writeflashpage(mtd, off, ctx->bounce);
 | 
						|
		if (rc2 <= 0) {
 | 
						|
			printk(KERN_EMERG
 | 
						|
			       "apanic: Flash write failed (%d)\n", rc2);
 | 
						|
			return idx;
 | 
						|
		}
 | 
						|
		if (!last_chunk)
 | 
						|
			idx += rc2;
 | 
						|
		else
 | 
						|
			idx += last_chunk;
 | 
						|
		off += rc2;
 | 
						|
	}
 | 
						|
	return idx;
 | 
						|
}
 | 
						|
 | 
						|
static int apanic(struct notifier_block *this, unsigned long event,
 | 
						|
			void *ptr)
 | 
						|
{
 | 
						|
	struct apanic_data *ctx = &drv_ctx;
 | 
						|
	struct panic_header *hdr = (struct panic_header *) ctx->bounce;
 | 
						|
	int console_offset = 0;
 | 
						|
	int console_len = 0;
 | 
						|
	int threads_offset = 0;
 | 
						|
	int threads_len = 0;
 | 
						|
	int rc;
 | 
						|
 | 
						|
	if (in_panic)
 | 
						|
		return NOTIFY_DONE;
 | 
						|
	in_panic = 1;
 | 
						|
#ifdef CONFIG_PREEMPT
 | 
						|
	/* Ensure that cond_resched() won't try to preempt anybody */
 | 
						|
	add_preempt_count(PREEMPT_ACTIVE);
 | 
						|
#endif
 | 
						|
	touch_softlockup_watchdog();
 | 
						|
 | 
						|
	if (!ctx->mtd)
 | 
						|
		goto out;
 | 
						|
 | 
						|
	if (ctx->curr.magic) {
 | 
						|
		printk(KERN_EMERG "Crash partition in use!\n");
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
	console_offset = ctx->mtd->writesize;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Write out the console
 | 
						|
	 */
 | 
						|
	console_len = apanic_write_console(ctx->mtd, console_offset);
 | 
						|
	if (console_len < 0) {
 | 
						|
		printk(KERN_EMERG "Error writing console to panic log! (%d)\n",
 | 
						|
		       console_len);
 | 
						|
		console_len = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Write out all threads
 | 
						|
	 */
 | 
						|
	threads_offset = ALIGN(console_offset + console_len,
 | 
						|
			       ctx->mtd->writesize);
 | 
						|
	if (!threads_offset)
 | 
						|
		threads_offset = ctx->mtd->writesize;
 | 
						|
 | 
						|
	ram_console_enable_console(0);
 | 
						|
 | 
						|
	log_buf_clear();
 | 
						|
	show_state_filter(0);
 | 
						|
	threads_len = apanic_write_console(ctx->mtd, threads_offset);
 | 
						|
	if (threads_len < 0) {
 | 
						|
		printk(KERN_EMERG "Error writing threads to panic log! (%d)\n",
 | 
						|
		       threads_len);
 | 
						|
		threads_len = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Finally write the panic header
 | 
						|
	 */
 | 
						|
	memset(ctx->bounce, 0, PAGE_SIZE);
 | 
						|
	hdr->magic = PANIC_MAGIC;
 | 
						|
	hdr->version = PHDR_VERSION;
 | 
						|
 | 
						|
	hdr->console_offset = console_offset;
 | 
						|
	hdr->console_length = console_len;
 | 
						|
 | 
						|
	hdr->threads_offset = threads_offset;
 | 
						|
	hdr->threads_length = threads_len;
 | 
						|
 | 
						|
	rc = apanic_writeflashpage(ctx->mtd, 0, ctx->bounce);
 | 
						|
	if (rc <= 0) {
 | 
						|
		printk(KERN_EMERG "apanic: Header write failed (%d)\n",
 | 
						|
		       rc);
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	printk(KERN_EMERG "apanic: Panic dump sucessfully written to flash\n");
 | 
						|
 | 
						|
 out:
 | 
						|
#ifdef CONFIG_PREEMPT
 | 
						|
	sub_preempt_count(PREEMPT_ACTIVE);
 | 
						|
#endif
 | 
						|
	in_panic = 0;
 | 
						|
	return NOTIFY_DONE;
 | 
						|
}
 | 
						|
 | 
						|
static struct notifier_block panic_blk = {
 | 
						|
	.notifier_call	= apanic,
 | 
						|
};
 | 
						|
 | 
						|
static int panic_dbg_get(void *data, u64 *val)
 | 
						|
{
 | 
						|
	apanic(NULL, 0, NULL);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int panic_dbg_set(void *data, u64 val)
 | 
						|
{
 | 
						|
	BUG();
 | 
						|
	return -1;
 | 
						|
}
 | 
						|
 | 
						|
DEFINE_SIMPLE_ATTRIBUTE(panic_dbg_fops, panic_dbg_get, panic_dbg_set, "%llu\n");
 | 
						|
 | 
						|
int __init apanic_init(void)
 | 
						|
{
 | 
						|
	register_mtd_user(&mtd_panic_notifier);
 | 
						|
	atomic_notifier_chain_register(&panic_notifier_list, &panic_blk);
 | 
						|
	debugfs_create_file("apanic", 0644, NULL, NULL, &panic_dbg_fops);
 | 
						|
	memset(&drv_ctx, 0, sizeof(drv_ctx));
 | 
						|
	drv_ctx.bounce = (void *) __get_free_page(GFP_KERNEL);
 | 
						|
	INIT_WORK(&proc_removal_work, apanic_remove_proc_work);
 | 
						|
	printk(KERN_INFO "Android kernel panic handler initialized (bind=%s)\n",
 | 
						|
	       CONFIG_APANIC_PLABEL);
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
module_init(apanic_init);
 |