umem.c 29.8 KB
Newer Older
1
// SPDX-License-Identifier: GPL-2.0-only
L
Linus Torvalds 已提交
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
/*
 * mm.c - Micro Memory(tm) PCI memory board block device driver - v2.3
 *
 * (C) 2001 San Mehat <nettwerk@valinux.com>
 * (C) 2001 Johannes Erdfelt <jerdfelt@valinux.com>
 * (C) 2001 NeilBrown <neilb@cse.unsw.edu.au>
 *
 * This driver for the Micro Memory PCI Memory Module with Battery Backup
 * is Copyright Micro Memory Inc 2001-2002.  All rights reserved.
 *
 * This driver provides a standard block device interface for Micro Memory(tm)
 * PCI based RAM boards.
 * 10/05/01: Phap Nguyen - Rebuilt the driver
 * 10/22/01: Phap Nguyen - v2.1 Added disk partitioning
 * 29oct2001:NeilBrown   - Use make_request_fn instead of request_fn
 *                       - use stand disk partitioning (so fdisk works).
 * 08nov2001:NeilBrown	 - change driver name from "mm" to "umem"
 *			 - incorporate into main kernel
 * 08apr2002:NeilBrown   - Move some of interrupt handle to tasklet
 *			 - use spin_lock_bh instead of _irq
 *			 - Never block on make_request.  queue
 *			   bh's instead.
 *			 - unregister umem from devfs at mod unload
 *			 - Change version to 2.3
 * 07Nov2001:Phap Nguyen - Select pci read command: 06, 12, 15 (Decimal)
 * 07Jan2002: P. Nguyen  - Used PCI Memory Write & Invalidate for DMA
 * 15May2002:NeilBrown   - convert to bio for 2.5
 * 17May2002:NeilBrown   - remove init_mem initialisation.  Instead detect
 *			 - a sequence of writes that cover the card, and
 *			 - set initialised bit then.
 */

34
#undef DEBUG	/* #define DEBUG if you want debugging info (pr_debug) */
L
Linus Torvalds 已提交
35 36 37 38 39
#include <linux/fs.h>
#include <linux/bio.h>
#include <linux/kernel.h>
#include <linux/mm.h>
#include <linux/mman.h>
40
#include <linux/gfp.h>
L
Linus Torvalds 已提交
41 42 43 44 45 46
#include <linux/ioctl.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/timer.h>
#include <linux/pci.h>
47
#include <linux/dma-mapping.h>
L
Linus Torvalds 已提交
48 49 50 51

#include <linux/fcntl.h>        /* O_ACCMODE */
#include <linux/hdreg.h>  /* HDIO_GETGEO */

52
#include "umem.h"
L
Linus Torvalds 已提交
53

54
#include <linux/uaccess.h>
L
Linus Torvalds 已提交
55 56 57 58 59 60 61 62 63 64 65 66
#include <asm/io.h>

#define MM_MAXCARDS 4
#define MM_RAHEAD 2      /* two sectors */
#define MM_BLKSIZE 1024  /* 1k blocks */
#define MM_HARDSECT 512  /* 512-byte hardware sectors */
#define MM_SHIFT 6       /* max 64 partitions on 4 cards  */

/*
 * Version Information
 */

J
Jeff Garzik 已提交
67 68 69 70
#define DRIVER_NAME	"umem"
#define DRIVER_VERSION	"v2.3"
#define DRIVER_AUTHOR	"San Mehat, Johannes Erdfelt, NeilBrown"
#define DRIVER_DESC	"Micro Memory(tm) PCI memory board block driver"
L
Linus Torvalds 已提交
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107

static int debug;
/* #define HW_TRACE(x)     writeb(x,cards[0].csr_remap + MEMCTRLSTATUS_MAGIC) */
#define HW_TRACE(x)

#define DEBUG_LED_ON_TRANSFER	0x01
#define DEBUG_BATTERY_POLLING	0x02

module_param(debug, int, 0644);
MODULE_PARM_DESC(debug, "Debug bitmask");

static int pci_read_cmd = 0x0C;		/* Read Multiple */
module_param(pci_read_cmd, int, 0);
MODULE_PARM_DESC(pci_read_cmd, "PCI read command");

static int pci_write_cmd = 0x0F;	/* Write and Invalidate */
module_param(pci_write_cmd, int, 0);
MODULE_PARM_DESC(pci_write_cmd, "PCI write command");

static int pci_cmds;

static int major_nr;

#include <linux/blkdev.h>
#include <linux/blkpg.h>

struct cardinfo {
	struct pci_dev	*dev;

	unsigned char	__iomem *csr_remap;
	unsigned int	mm_size;  /* size in kbytes */

	unsigned int	init_size; /* initial segment, in sectors,
				    * that we know to
				    * have been written
				    */
	struct bio	*bio, *currentbio, **biotail;
108
	struct bvec_iter current_iter;
L
Linus Torvalds 已提交
109

110
	struct request_queue *queue;
L
Linus Torvalds 已提交
111 112 113 114 115 116

	struct mm_page {
		dma_addr_t		page_dma;
		struct mm_dma_desc	*desc;
		int	 		cnt, headcnt;
		struct bio		*bio, **biotail;
117
		struct bvec_iter	iter;
L
Linus Torvalds 已提交
118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
	} mm_pages[2];
#define DESC_PER_PAGE ((PAGE_SIZE*2)/sizeof(struct mm_dma_desc))

	int  Active, Ready;

	struct tasklet_struct	tasklet;
	unsigned int dma_status;

	struct {
		int		good;
		int		warned;
		unsigned long	last_change;
	} battery[2];

	spinlock_t 	lock;
	int		check_batteries;

	int		flags;
};

static struct cardinfo cards[MM_MAXCARDS];
static struct timer_list battery_timer;

141
static int num_cards;
L
Linus Torvalds 已提交
142 143 144 145 146 147 148 149 150 151 152 153

static struct gendisk *mm_gendisk[MM_MAXCARDS];

static void check_batteries(struct cardinfo *card);

static int get_userbit(struct cardinfo *card, int bit)
{
	unsigned char led;

	led = readb(card->csr_remap + MEMCTRLCMD_LEDCTRL);
	return led & bit;
}
154

L
Linus Torvalds 已提交
155 156 157 158 159 160 161 162 163 164 165 166 167
static int set_userbit(struct cardinfo *card, int bit, unsigned char state)
{
	unsigned char led;

	led = readb(card->csr_remap + MEMCTRLCMD_LEDCTRL);
	if (state)
		led |= bit;
	else
		led &= ~bit;
	writeb(led, card->csr_remap + MEMCTRLCMD_LEDCTRL);

	return 0;
}
168

L
Linus Torvalds 已提交
169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203
/*
 * NOTE: For the power LED, use the LED_POWER_* macros since they differ
 */
static void set_led(struct cardinfo *card, int shift, unsigned char state)
{
	unsigned char led;

	led = readb(card->csr_remap + MEMCTRLCMD_LEDCTRL);
	if (state == LED_FLIP)
		led ^= (1<<shift);
	else {
		led &= ~(0x03 << shift);
		led |= (state << shift);
	}
	writeb(led, card->csr_remap + MEMCTRLCMD_LEDCTRL);

}

#ifdef MM_DIAG
static void dump_regs(struct cardinfo *card)
{
	unsigned char *p;
	int i, i1;

	p = card->csr_remap;
	for (i = 0; i < 8; i++) {
		printk(KERN_DEBUG "%p   ", p);

		for (i1 = 0; i1 < 16; i1++)
			printk("%02x ", *p++);

		printk("\n");
	}
}
#endif
204

L
Linus Torvalds 已提交
205 206
static void dump_dmastat(struct cardinfo *card, unsigned int dmastat)
{
J
Jeff Garzik 已提交
207
	dev_printk(KERN_DEBUG, &card->dev->dev, "DMAstat - ");
L
Linus Torvalds 已提交
208
	if (dmastat & DMASCR_ANY_ERR)
209
		printk(KERN_CONT "ANY_ERR ");
L
Linus Torvalds 已提交
210
	if (dmastat & DMASCR_MBE_ERR)
211
		printk(KERN_CONT "MBE_ERR ");
L
Linus Torvalds 已提交
212
	if (dmastat & DMASCR_PARITY_ERR_REP)
213
		printk(KERN_CONT "PARITY_ERR_REP ");
L
Linus Torvalds 已提交
214
	if (dmastat & DMASCR_PARITY_ERR_DET)
215
		printk(KERN_CONT "PARITY_ERR_DET ");
L
Linus Torvalds 已提交
216
	if (dmastat & DMASCR_SYSTEM_ERR_SIG)
217
		printk(KERN_CONT "SYSTEM_ERR_SIG ");
L
Linus Torvalds 已提交
218
	if (dmastat & DMASCR_TARGET_ABT)
219
		printk(KERN_CONT "TARGET_ABT ");
L
Linus Torvalds 已提交
220
	if (dmastat & DMASCR_MASTER_ABT)
221
		printk(KERN_CONT "MASTER_ABT ");
L
Linus Torvalds 已提交
222
	if (dmastat & DMASCR_CHAIN_COMPLETE)
223
		printk(KERN_CONT "CHAIN_COMPLETE ");
L
Linus Torvalds 已提交
224
	if (dmastat & DMASCR_DMA_COMPLETE)
225
		printk(KERN_CONT "DMA_COMPLETE ");
L
Linus Torvalds 已提交
226 227 228 229 230 231 232 233 234 235 236 237 238 239
	printk("\n");
}

/*
 * Theory of request handling
 *
 * Each bio is assigned to one mm_dma_desc - which may not be enough FIXME
 * We have two pages of mm_dma_desc, holding about 64 descriptors
 * each.  These are allocated at init time.
 * One page is "Ready" and is either full, or can have request added.
 * The other page might be "Active", which DMA is happening on it.
 *
 * Whenever IO on the active page completes, the Ready page is activated
 * and the ex-Active page is clean out and made Ready.
J
Jens Axboe 已提交
240
 * Otherwise the Ready page is only activated when it becomes full.
L
Linus Torvalds 已提交
241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260
 *
 * If a request arrives while both pages a full, it is queued, and b_rdev is
 * overloaded to record whether it was a read or a write.
 *
 * The interrupt handler only polls the device to clear the interrupt.
 * The processing of the result is done in a tasklet.
 */

static void mm_start_io(struct cardinfo *card)
{
	/* we have the lock, we know there is
	 * no IO active, and we know that card->Active
	 * is set
	 */
	struct mm_dma_desc *desc;
	struct mm_page *page;
	int offset;

	/* make the last descriptor end the chain */
	page = &card->mm_pages[card->Active];
261 262
	pr_debug("start_io: %d %d->%d\n",
		card->Active, page->headcnt, page->cnt - 1);
L
Linus Torvalds 已提交
263 264 265 266 267 268
	desc = &page->desc[page->cnt-1];

	desc->control_bits |= cpu_to_le32(DMASCR_CHAIN_COMP_EN);
	desc->control_bits &= ~cpu_to_le32(DMASCR_CHAIN_EN);
	desc->sem_control_bits = desc->control_bits;

269

L
Linus Torvalds 已提交
270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285
	if (debug & DEBUG_LED_ON_TRANSFER)
		set_led(card, LED_REMOVE, LED_ON);

	desc = &page->desc[page->headcnt];
	writel(0, card->csr_remap + DMA_PCI_ADDR);
	writel(0, card->csr_remap + DMA_PCI_ADDR + 4);

	writel(0, card->csr_remap + DMA_LOCAL_ADDR);
	writel(0, card->csr_remap + DMA_LOCAL_ADDR + 4);

	writel(0, card->csr_remap + DMA_TRANSFER_SIZE);
	writel(0, card->csr_remap + DMA_TRANSFER_SIZE + 4);

	writel(0, card->csr_remap + DMA_SEMAPHORE_ADDR);
	writel(0, card->csr_remap + DMA_SEMAPHORE_ADDR + 4);

286 287
	offset = ((char *)desc) - ((char *)page->desc);
	writel(cpu_to_le32((page->page_dma+offset) & 0xffffffff),
L
Linus Torvalds 已提交
288 289 290 291 292 293 294 295 296 297 298 299 300 301 302
	       card->csr_remap + DMA_DESCRIPTOR_ADDR);
	/* Force the value to u64 before shifting otherwise >> 32 is undefined C
	 * and on some ports will do nothing ! */
	writel(cpu_to_le32(((u64)page->page_dma)>>32),
	       card->csr_remap + DMA_DESCRIPTOR_ADDR + 4);

	/* Go, go, go */
	writel(cpu_to_le32(DMASCR_GO | DMASCR_CHAIN_EN | pci_cmds),
	       card->csr_remap + DMA_STATUS_CTRL);
}

static int add_bio(struct cardinfo *card);

static void activate(struct cardinfo *card)
{
303
	/* if No page is Active, and Ready is
L
Linus Torvalds 已提交
304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327
	 * not empty, then switch Ready page
	 * to active and start IO.
	 * Then add any bh's that are available to Ready
	 */

	do {
		while (add_bio(card))
			;

		if (card->Active == -1 &&
		    card->mm_pages[card->Ready].cnt > 0) {
			card->Active = card->Ready;
			card->Ready = 1-card->Ready;
			mm_start_io(card);
		}

	} while (card->Active == -1 && add_bio(card));
}

static inline void reset_page(struct mm_page *page)
{
	page->cnt = 0;
	page->headcnt = 0;
	page->bio = NULL;
328
	page->biotail = &page->bio;
L
Linus Torvalds 已提交
329 330
}

331
/*
L
Linus Torvalds 已提交
332 333 334 335 336 337 338 339 340 341 342
 * If there is room on Ready page, take
 * one bh off list and add it.
 * return 1 if there was room, else 0.
 */
static int add_bio(struct cardinfo *card)
{
	struct mm_page *p;
	struct mm_dma_desc *desc;
	dma_addr_t dma_handle;
	int offset;
	struct bio *bio;
343
	struct bio_vec vec;
L
Linus Torvalds 已提交
344 345 346 347

	bio = card->currentbio;
	if (!bio && card->bio) {
		card->currentbio = card->bio;
348
		card->current_iter = card->bio->bi_iter;
L
Linus Torvalds 已提交
349 350 351 352 353 354 355 356 357 358 359 360
		card->bio = card->bio->bi_next;
		if (card->bio == NULL)
			card->biotail = &card->bio;
		card->currentbio->bi_next = NULL;
		return 1;
	}
	if (!bio)
		return 0;

	if (card->mm_pages[card->Ready].cnt >= DESC_PER_PAGE)
		return 0;

361 362
	vec = bio_iter_iovec(bio, card->current_iter);

363
	dma_handle = dma_map_page(&card->dev->dev,
364 365 366
				  vec.bv_page,
				  vec.bv_offset,
				  vec.bv_len,
367
				  bio_op(bio) == REQ_OP_READ ?
368
				  DMA_FROM_DEVICE : DMA_TO_DEVICE);
L
Linus Torvalds 已提交
369 370 371 372

	p = &card->mm_pages[card->Ready];
	desc = &p->desc[p->cnt];
	p->cnt++;
373
	if (p->bio == NULL)
374
		p->iter = card->current_iter;
L
Linus Torvalds 已提交
375 376 377 378 379 380 381 382 383
	if ((p->biotail) != &bio->bi_next) {
		*(p->biotail) = bio;
		p->biotail = &(bio->bi_next);
		bio->bi_next = NULL;
	}

	desc->data_dma_handle = dma_handle;

	desc->pci_addr = cpu_to_le64((u64)desc->data_dma_handle);
384 385
	desc->local_addr = cpu_to_le64(card->current_iter.bi_sector << 9);
	desc->transfer_size = cpu_to_le32(vec.bv_len);
386
	offset = (((char *)&desc->sem_control_bits) - ((char *)p->desc));
L
Linus Torvalds 已提交
387 388
	desc->sem_addr = cpu_to_le64((u64)(p->page_dma+offset));
	desc->zero1 = desc->zero2 = 0;
389
	offset = (((char *)(desc+1)) - ((char *)p->desc));
L
Linus Torvalds 已提交
390 391 392 393 394 395
	desc->next_desc_addr = cpu_to_le64(p->page_dma+offset);
	desc->control_bits = cpu_to_le32(DMASCR_GO|DMASCR_ERR_INT_EN|
					 DMASCR_PARITY_INT_EN|
					 DMASCR_CHAIN_EN |
					 DMASCR_SEM_EN |
					 pci_cmds);
396
	if (bio_op(bio) == REQ_OP_WRITE)
L
Linus Torvalds 已提交
397 398 399
		desc->control_bits |= cpu_to_le32(DMASCR_TRANSFER_READ);
	desc->sem_control_bits = desc->control_bits;

400 401 402

	bio_advance_iter(bio, &card->current_iter, vec.bv_len);
	if (!card->current_iter.bi_size)
L
Linus Torvalds 已提交
403 404 405 406 407 408 409 410 411 412
		card->currentbio = NULL;

	return 1;
}

static void process_page(unsigned long data)
{
	/* check if any of the requests in the page are DMA_COMPLETE,
	 * and deal with them appropriately.
	 * If we find a descriptor without DMA_COMPLETE in the semaphore, then
413 414
	 * dma must have hit an error on that descriptor, so use dma_status
	 * instead and assume that all following descriptors must be re-tried.
L
Linus Torvalds 已提交
415 416
	 */
	struct mm_page *page;
417
	struct bio *return_bio = NULL;
L
Linus Torvalds 已提交
418 419 420
	struct cardinfo *card = (struct cardinfo *)data;
	unsigned int dma_status = card->dma_status;

421
	spin_lock(&card->lock);
L
Linus Torvalds 已提交
422 423 424
	if (card->Active < 0)
		goto out_unlock;
	page = &card->mm_pages[card->Active];
425

L
Linus Torvalds 已提交
426 427 428 429
	while (page->headcnt < page->cnt) {
		struct bio *bio = page->bio;
		struct mm_dma_desc *desc = &page->desc[page->headcnt];
		int control = le32_to_cpu(desc->sem_control_bits);
430
		int last = 0;
431
		struct bio_vec vec;
L
Linus Torvalds 已提交
432 433 434

		if (!(control & DMASCR_DMA_COMPLETE)) {
			control = dma_status;
435
			last = 1;
L
Linus Torvalds 已提交
436
		}
437

L
Linus Torvalds 已提交
438
		page->headcnt++;
439 440 441 442
		vec = bio_iter_iovec(bio, page->iter);
		bio_advance_iter(bio, &page->iter, vec.bv_len);

		if (!page->iter.bi_size) {
L
Linus Torvalds 已提交
443
			page->bio = bio->bi_next;
N
Neil Brown 已提交
444
			if (page->bio)
445
				page->iter = page->bio->bi_iter;
446
		}
L
Linus Torvalds 已提交
447

448
		dma_unmap_page(&card->dev->dev, desc->data_dma_handle,
449
			       vec.bv_len,
450
				 (control & DMASCR_TRANSFER_READ) ?
451
				DMA_TO_DEVICE : DMA_FROM_DEVICE);
L
Linus Torvalds 已提交
452 453
		if (control & DMASCR_HARD_ERROR) {
			/* error */
454
			bio->bi_status = BLK_STS_IOERR;
J
Jeff Garzik 已提交
455 456 457 458
			dev_printk(KERN_WARNING, &card->dev->dev,
				"I/O error on sector %d/%d\n",
				le32_to_cpu(desc->local_addr)>>9,
				le32_to_cpu(desc->transfer_size));
L
Linus Torvalds 已提交
459
			dump_dmastat(card, control);
460
		} else if (op_is_write(bio_op(bio)) &&
461 462 463 464
			   le32_to_cpu(desc->local_addr) >> 9 ==
				card->init_size) {
			card->init_size += le32_to_cpu(desc->transfer_size) >> 9;
			if (card->init_size >> 1 >= card->mm_size) {
J
Jeff Garzik 已提交
465 466
				dev_printk(KERN_INFO, &card->dev->dev,
					"memory now initialised\n");
L
Linus Torvalds 已提交
467 468 469 470 471 472 473 474
				set_userbit(card, MEMORY_INITIALIZED, 1);
			}
		}
		if (bio != page->bio) {
			bio->bi_next = return_bio;
			return_bio = bio;
		}

475 476
		if (last)
			break;
L
Linus Torvalds 已提交
477 478 479 480 481 482 483 484 485 486 487 488 489 490 491
	}

	if (debug & DEBUG_LED_ON_TRANSFER)
		set_led(card, LED_REMOVE, LED_OFF);

	if (card->check_batteries) {
		card->check_batteries = 0;
		check_batteries(card);
	}
	if (page->headcnt >= page->cnt) {
		reset_page(page);
		card->Active = -1;
		activate(card);
	} else {
		/* haven't finished with this one yet */
492
		pr_debug("do some more\n");
L
Linus Torvalds 已提交
493 494 495
		mm_start_io(card);
	}
 out_unlock:
496
	spin_unlock(&card->lock);
L
Linus Torvalds 已提交
497

498
	while (return_bio) {
L
Linus Torvalds 已提交
499 500 501 502
		struct bio *bio = return_bio;

		return_bio = bio->bi_next;
		bio->bi_next = NULL;
503
		bio_endio(bio);
L
Linus Torvalds 已提交
504 505 506
	}
}

507
static void mm_unplug(struct blk_plug_cb *cb, bool from_schedule)
T
Tao Guo 已提交
508
{
509
	struct cardinfo *card = cb->data;
T
Tao Guo 已提交
510

511 512 513 514
	spin_lock_irq(&card->lock);
	activate(card);
	spin_unlock_irq(&card->lock);
	kfree(cb);
T
Tao Guo 已提交
515 516 517 518
}

static int mm_check_plugged(struct cardinfo *card)
{
519
	return !!blk_check_plugged(mm_unplug, card, sizeof(struct blk_plug_cb));
T
Tao Guo 已提交
520 521
}

522
static blk_qc_t mm_make_request(struct request_queue *q, struct bio *bio)
L
Linus Torvalds 已提交
523 524
{
	struct cardinfo *card = q->queuedata;
525
	pr_debug("mm_make_request %llu %u\n",
526 527
		 (unsigned long long)bio->bi_iter.bi_sector,
		 bio->bi_iter.bi_size);
L
Linus Torvalds 已提交
528

529
	blk_queue_split(q, &bio);
530

L
Linus Torvalds 已提交
531 532 533 534
	spin_lock_irq(&card->lock);
	*card->biotail = bio;
	bio->bi_next = NULL;
	card->biotail = &bio->bi_next;
535
	if (op_is_sync(bio->bi_opf) || !mm_check_plugged(card))
T
Tao Guo 已提交
536
		activate(card);
L
Linus Torvalds 已提交
537 538
	spin_unlock_irq(&card->lock);

539
	return BLK_QC_T_NONE;
L
Linus Torvalds 已提交
540 541
}

542
static irqreturn_t mm_interrupt(int irq, void *__card)
L
Linus Torvalds 已提交
543 544 545 546 547 548 549 550 551 552 553 554
{
	struct cardinfo *card = (struct cardinfo *) __card;
	unsigned int dma_status;
	unsigned short cfg_status;

HW_TRACE(0x30);

	dma_status = le32_to_cpu(readl(card->csr_remap + DMA_STATUS_CTRL));

	if (!(dma_status & (DMASCR_ERROR_MASK | DMASCR_CHAIN_COMPLETE))) {
		/* interrupt wasn't for me ... */
		return IRQ_NONE;
555
	}
L
Linus Torvalds 已提交
556 557 558 559

	/* clear COMPLETION interrupts */
	if (card->flags & UM_FLAG_NO_BYTE_STATUS)
		writel(cpu_to_le32(DMASCR_DMA_COMPLETE|DMASCR_CHAIN_COMPLETE),
560
		       card->csr_remap + DMA_STATUS_CTRL);
L
Linus Torvalds 已提交
561 562
	else
		writeb((DMASCR_DMA_COMPLETE|DMASCR_CHAIN_COMPLETE) >> 16,
563
		       card->csr_remap + DMA_STATUS_CTRL + 2);
564

L
Linus Torvalds 已提交
565 566 567 568 569 570 571 572
	/* log errors and clear interrupt status */
	if (dma_status & DMASCR_ANY_ERR) {
		unsigned int	data_log1, data_log2;
		unsigned int	addr_log1, addr_log2;
		unsigned char	stat, count, syndrome, check;

		stat = readb(card->csr_remap + MEMCTRLCMD_ERRSTATUS);

573 574 575 576 577 578
		data_log1 = le32_to_cpu(readl(card->csr_remap +
						ERROR_DATA_LOG));
		data_log2 = le32_to_cpu(readl(card->csr_remap +
						ERROR_DATA_LOG + 4));
		addr_log1 = le32_to_cpu(readl(card->csr_remap +
						ERROR_ADDR_LOG));
L
Linus Torvalds 已提交
579 580 581 582 583 584 585 586 587
		addr_log2 = readb(card->csr_remap + ERROR_ADDR_LOG + 4);

		count = readb(card->csr_remap + ERROR_COUNT);
		syndrome = readb(card->csr_remap + ERROR_SYNDROME);
		check = readb(card->csr_remap + ERROR_CHECK);

		dump_dmastat(card, dma_status);

		if (stat & 0x01)
J
Jeff Garzik 已提交
588 589 590
			dev_printk(KERN_ERR, &card->dev->dev,
				"Memory access error detected (err count %d)\n",
				count);
L
Linus Torvalds 已提交
591
		if (stat & 0x02)
J
Jeff Garzik 已提交
592 593
			dev_printk(KERN_ERR, &card->dev->dev,
				"Multi-bit EDC error\n");
L
Linus Torvalds 已提交
594

J
Jeff Garzik 已提交
595 596 597 598 599 600
		dev_printk(KERN_ERR, &card->dev->dev,
			"Fault Address 0x%02x%08x, Fault Data 0x%08x%08x\n",
			addr_log2, addr_log1, data_log2, data_log1);
		dev_printk(KERN_ERR, &card->dev->dev,
			"Fault Check 0x%02x, Fault Syndrome 0x%02x\n",
			check, syndrome);
L
Linus Torvalds 已提交
601 602 603 604 605

		writeb(0, card->csr_remap + ERROR_COUNT);
	}

	if (dma_status & DMASCR_PARITY_ERR_REP) {
J
Jeff Garzik 已提交
606 607
		dev_printk(KERN_ERR, &card->dev->dev,
			"PARITY ERROR REPORTED\n");
L
Linus Torvalds 已提交
608 609 610 611 612
		pci_read_config_word(card->dev, PCI_STATUS, &cfg_status);
		pci_write_config_word(card->dev, PCI_STATUS, cfg_status);
	}

	if (dma_status & DMASCR_PARITY_ERR_DET) {
J
Jeff Garzik 已提交
613 614
		dev_printk(KERN_ERR, &card->dev->dev,
			"PARITY ERROR DETECTED\n");
L
Linus Torvalds 已提交
615 616 617 618 619
		pci_read_config_word(card->dev, PCI_STATUS, &cfg_status);
		pci_write_config_word(card->dev, PCI_STATUS, cfg_status);
	}

	if (dma_status & DMASCR_SYSTEM_ERR_SIG) {
J
Jeff Garzik 已提交
620
		dev_printk(KERN_ERR, &card->dev->dev, "SYSTEM ERROR\n");
L
Linus Torvalds 已提交
621 622 623 624 625
		pci_read_config_word(card->dev, PCI_STATUS, &cfg_status);
		pci_write_config_word(card->dev, PCI_STATUS, cfg_status);
	}

	if (dma_status & DMASCR_TARGET_ABT) {
J
Jeff Garzik 已提交
626
		dev_printk(KERN_ERR, &card->dev->dev, "TARGET ABORT\n");
L
Linus Torvalds 已提交
627 628 629 630 631
		pci_read_config_word(card->dev, PCI_STATUS, &cfg_status);
		pci_write_config_word(card->dev, PCI_STATUS, cfg_status);
	}

	if (dma_status & DMASCR_MASTER_ABT) {
J
Jeff Garzik 已提交
632
		dev_printk(KERN_ERR, &card->dev->dev, "MASTER ABORT\n");
L
Linus Torvalds 已提交
633 634 635 636 637 638 639 640 641 642
		pci_read_config_word(card->dev, PCI_STATUS, &cfg_status);
		pci_write_config_word(card->dev, PCI_STATUS, cfg_status);
	}

	/* and process the DMA descriptors */
	card->dma_status = dma_status;
	tasklet_schedule(&card->tasklet);

HW_TRACE(0x36);

643
	return IRQ_HANDLED;
L
Linus Torvalds 已提交
644
}
645

L
Linus Torvalds 已提交
646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672
/*
 * If both batteries are good, no LED
 * If either battery has been warned, solid LED
 * If both batteries are bad, flash the LED quickly
 * If either battery is bad, flash the LED semi quickly
 */
static void set_fault_to_battery_status(struct cardinfo *card)
{
	if (card->battery[0].good && card->battery[1].good)
		set_led(card, LED_FAULT, LED_OFF);
	else if (card->battery[0].warned || card->battery[1].warned)
		set_led(card, LED_FAULT, LED_ON);
	else if (!card->battery[0].good && !card->battery[1].good)
		set_led(card, LED_FAULT, LED_FLASH_7_0);
	else
		set_led(card, LED_FAULT, LED_FLASH_3_5);
}

static void init_battery_timer(void);

static int check_battery(struct cardinfo *card, int battery, int status)
{
	if (status != card->battery[battery].good) {
		card->battery[battery].good = !card->battery[battery].good;
		card->battery[battery].last_change = jiffies;

		if (card->battery[battery].good) {
J
Jeff Garzik 已提交
673 674
			dev_printk(KERN_ERR, &card->dev->dev,
				"Battery %d now good\n", battery + 1);
L
Linus Torvalds 已提交
675 676
			card->battery[battery].warned = 0;
		} else
J
Jeff Garzik 已提交
677 678
			dev_printk(KERN_ERR, &card->dev->dev,
				"Battery %d now FAILED\n", battery + 1);
L
Linus Torvalds 已提交
679 680 681 682 683 684

		return 1;
	} else if (!card->battery[battery].good &&
		   !card->battery[battery].warned &&
		   time_after_eq(jiffies, card->battery[battery].last_change +
				 (HZ * 60 * 60 * 5))) {
J
Jeff Garzik 已提交
685 686
		dev_printk(KERN_ERR, &card->dev->dev,
			"Battery %d still FAILED after 5 hours\n", battery + 1);
L
Linus Torvalds 已提交
687 688 689 690 691 692 693
		card->battery[battery].warned = 1;

		return 1;
	}

	return 0;
}
694

L
Linus Torvalds 已提交
695 696 697 698 699 700 701 702 703 704 705
static void check_batteries(struct cardinfo *card)
{
	/* NOTE: this must *never* be called while the card
	 * is doing (bus-to-card) DMA, or you will need the
	 * reset switch
	 */
	unsigned char status;
	int ret1, ret2;

	status = readb(card->csr_remap + MEMCTRLSTATUS_BATTERY);
	if (debug & DEBUG_BATTERY_POLLING)
J
Jeff Garzik 已提交
706 707
		dev_printk(KERN_DEBUG, &card->dev->dev,
			"checking battery status, 1 = %s, 2 = %s\n",
L
Linus Torvalds 已提交
708 709 710 711 712 713 714 715 716 717
		       (status & BATTERY_1_FAILURE) ? "FAILURE" : "OK",
		       (status & BATTERY_2_FAILURE) ? "FAILURE" : "OK");

	ret1 = check_battery(card, 0, !(status & BATTERY_1_FAILURE));
	ret2 = check_battery(card, 1, !(status & BATTERY_2_FAILURE));

	if (ret1 || ret2)
		set_fault_to_battery_status(card);
}

718
static void check_all_batteries(struct timer_list *unused)
L
Linus Torvalds 已提交
719 720 721
{
	int i;

722
	for (i = 0; i < num_cards; i++)
L
Linus Torvalds 已提交
723 724 725 726 727 728 729 730 731 732 733 734
		if (!(cards[i].flags & UM_FLAG_NO_BATT)) {
			struct cardinfo *card = &cards[i];
			spin_lock_bh(&card->lock);
			if (card->Active >= 0)
				card->check_batteries = 1;
			else
				check_batteries(card);
			spin_unlock_bh(&card->lock);
		}

	init_battery_timer();
}
735

L
Linus Torvalds 已提交
736 737
static void init_battery_timer(void)
{
738
	timer_setup(&battery_timer, check_all_batteries, 0);
L
Linus Torvalds 已提交
739 740 741
	battery_timer.expires = jiffies + (HZ * 60);
	add_timer(&battery_timer);
}
742

L
Linus Torvalds 已提交
743 744 745 746
static void del_battery_timer(void)
{
	del_timer(&battery_timer);
}
747

L
Linus Torvalds 已提交
748 749 750 751 752 753 754 755 756 757 758 759 760 761
/*
 * Note no locks taken out here.  In a worst case scenario, we could drop
 * a chunk of system memory.  But that should never happen, since validation
 * happens at open or mount time, when locks are held.
 *
 *	That's crap, since doing that while some partitions are opened
 * or mounted will give you really nasty results.
 */
static int mm_revalidate(struct gendisk *disk)
{
	struct cardinfo *card = disk->private_data;
	set_capacity(disk, card->mm_size << 1);
	return 0;
}
762 763

static int mm_getgeo(struct block_device *bdev, struct hd_geometry *geo)
L
Linus Torvalds 已提交
764
{
765 766
	struct cardinfo *card = bdev->bd_disk->private_data;
	int size = card->mm_size * (1024 / MM_HARDSECT);
L
Linus Torvalds 已提交
767

768 769 770 771 772 773 774 775 776
	/*
	 * get geometry: we have to fake one...  trim the size to a
	 * multiple of 2048 (1M): tell we have 32 sectors, 64 heads,
	 * whatever cylinders.
	 */
	geo->heads     = 64;
	geo->sectors   = 32;
	geo->cylinders = size / (geo->heads * geo->sectors);
	return 0;
L
Linus Torvalds 已提交
777
}
778

779
static const struct block_device_operations mm_fops = {
L
Linus Torvalds 已提交
780
	.owner		= THIS_MODULE,
781
	.getgeo		= mm_getgeo,
782
	.revalidate_disk = mm_revalidate,
L
Linus Torvalds 已提交
783
};
784

785
static int mm_pci_probe(struct pci_dev *dev, const struct pci_device_id *id)
L
Linus Torvalds 已提交
786 787 788 789 790 791
{
	int ret = -ENODEV;
	struct cardinfo *card = &cards[num_cards];
	unsigned char	mem_present;
	unsigned char	batt_status;
	unsigned int	saved_bar, data;
J
Jeff Garzik 已提交
792 793
	unsigned long	csr_base;
	unsigned long	csr_len;
L
Linus Torvalds 已提交
794
	int		magic_number;
J
Jeff Garzik 已提交
795 796 797 798
	static int	printed_version;

	if (!printed_version++)
		printk(KERN_INFO DRIVER_VERSION " : " DRIVER_DESC "\n");
L
Linus Torvalds 已提交
799

J
Jeff Garzik 已提交
800 801 802
	ret = pci_enable_device(dev);
	if (ret)
		return ret;
L
Linus Torvalds 已提交
803 804 805 806 807 808

	pci_write_config_byte(dev, PCI_LATENCY_TIMER, 0xF8);
	pci_set_master(dev);

	card->dev         = dev;

J
Jeff Garzik 已提交
809 810 811 812
	csr_base = pci_resource_start(dev, 0);
	csr_len  = pci_resource_len(dev, 0);
	if (!csr_base || !csr_len)
		return -ENODEV;
L
Linus Torvalds 已提交
813

J
Jeff Garzik 已提交
814
	dev_printk(KERN_INFO, &dev->dev,
815
	  "Micro Memory(tm) controller found (PCI Mem Module (Battery Backup))\n");
L
Linus Torvalds 已提交
816

817 818
	if (dma_set_mask(&dev->dev, DMA_BIT_MASK(64)) &&
	    dma_set_mask(&dev->dev, DMA_BIT_MASK(32))) {
J
Jeff Garzik 已提交
819
		dev_printk(KERN_WARNING, &dev->dev, "NO suitable DMA found\n");
L
Linus Torvalds 已提交
820 821
		return  -ENOMEM;
	}
J
Jeff Garzik 已提交
822 823 824

	ret = pci_request_regions(dev, DRIVER_NAME);
	if (ret) {
J
Jeff Garzik 已提交
825 826
		dev_printk(KERN_ERR, &card->dev->dev,
			"Unable to request memory region\n");
L
Linus Torvalds 已提交
827 828 829
		goto failed_req_csr;
	}

830
	card->csr_remap = ioremap(csr_base, csr_len);
L
Linus Torvalds 已提交
831
	if (!card->csr_remap) {
J
Jeff Garzik 已提交
832 833
		dev_printk(KERN_ERR, &card->dev->dev,
			"Unable to remap memory region\n");
L
Linus Torvalds 已提交
834 835 836 837 838
		ret = -ENOMEM;

		goto failed_remap_csr;
	}

J
Jeff Garzik 已提交
839 840
	dev_printk(KERN_INFO, &card->dev->dev,
		"CSR 0x%08lx -> 0x%p (0x%lx)\n",
J
Jeff Garzik 已提交
841
	       csr_base, card->csr_remap, csr_len);
L
Linus Torvalds 已提交
842

843
	switch (card->dev->device) {
L
Linus Torvalds 已提交
844 845 846 847 848 849 850 851 852 853 854
	case 0x5415:
		card->flags |= UM_FLAG_NO_BYTE_STATUS | UM_FLAG_NO_BATTREG;
		magic_number = 0x59;
		break;

	case 0x5425:
		card->flags |= UM_FLAG_NO_BYTE_STATUS;
		magic_number = 0x5C;
		break;

	case 0x6155:
855 856
		card->flags |= UM_FLAG_NO_BYTE_STATUS |
				UM_FLAG_NO_BATTREG | UM_FLAG_NO_BATT;
L
Linus Torvalds 已提交
857 858 859 860 861 862 863 864 865
		magic_number = 0x99;
		break;

	default:
		magic_number = 0x100;
		break;
	}

	if (readb(card->csr_remap + MEMCTRLSTATUS_MAGIC) != magic_number) {
J
Jeff Garzik 已提交
866
		dev_printk(KERN_ERR, &card->dev->dev, "Magic number invalid\n");
L
Linus Torvalds 已提交
867 868 869 870
		ret = -ENOMEM;
		goto failed_magic;
	}

871 872 873 874
	card->mm_pages[0].desc = dma_alloc_coherent(&card->dev->dev,
			PAGE_SIZE * 2, &card->mm_pages[0].page_dma, GFP_KERNEL);
	card->mm_pages[1].desc = dma_alloc_coherent(&card->dev->dev,
			PAGE_SIZE * 2, &card->mm_pages[1].page_dma, GFP_KERNEL);
L
Linus Torvalds 已提交
875 876
	if (card->mm_pages[0].desc == NULL ||
	    card->mm_pages[1].desc == NULL) {
J
Jeff Garzik 已提交
877
		dev_printk(KERN_ERR, &card->dev->dev, "alloc failed\n");
L
Linus Torvalds 已提交
878 879 880 881 882 883 884 885
		goto failed_alloc;
	}
	reset_page(&card->mm_pages[0]);
	reset_page(&card->mm_pages[1]);
	card->Ready = 0;	/* page 0 is ready */
	card->Active = -1;	/* no page is active */
	card->bio = NULL;
	card->biotail = &card->bio;
886
	spin_lock_init(&card->lock);
L
Linus Torvalds 已提交
887

888
	card->queue = blk_alloc_queue_node(GFP_KERNEL, NUMA_NO_NODE);
L
Linus Torvalds 已提交
889 890 891 892 893 894 895 896 897
	if (!card->queue)
		goto failed_alloc;

	blk_queue_make_request(card->queue, mm_make_request);
	card->queue->queuedata = card;

	tasklet_init(&card->tasklet, process_page, (unsigned long)card);

	card->check_batteries = 0;
898

L
Linus Torvalds 已提交
899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930
	mem_present = readb(card->csr_remap + MEMCTRLSTATUS_MEMORY);
	switch (mem_present) {
	case MEM_128_MB:
		card->mm_size = 1024 * 128;
		break;
	case MEM_256_MB:
		card->mm_size = 1024 * 256;
		break;
	case MEM_512_MB:
		card->mm_size = 1024 * 512;
		break;
	case MEM_1_GB:
		card->mm_size = 1024 * 1024;
		break;
	case MEM_2_GB:
		card->mm_size = 1024 * 2048;
		break;
	default:
		card->mm_size = 0;
		break;
	}

	/* Clear the LED's we control */
	set_led(card, LED_REMOVE, LED_OFF);
	set_led(card, LED_FAULT, LED_OFF);

	batt_status = readb(card->csr_remap + MEMCTRLSTATUS_BATTERY);

	card->battery[0].good = !(batt_status & BATTERY_1_FAILURE);
	card->battery[1].good = !(batt_status & BATTERY_2_FAILURE);
	card->battery[0].last_change = card->battery[1].last_change = jiffies;

931
	if (card->flags & UM_FLAG_NO_BATT)
J
Jeff Garzik 已提交
932 933
		dev_printk(KERN_INFO, &card->dev->dev,
			"Size %d KB\n", card->mm_size);
L
Linus Torvalds 已提交
934
	else {
J
Jeff Garzik 已提交
935 936 937
		dev_printk(KERN_INFO, &card->dev->dev,
			"Size %d KB, Battery 1 %s (%s), Battery 2 %s (%s)\n",
		       card->mm_size,
938
		       batt_status & BATTERY_1_DISABLED ? "Disabled" : "Enabled",
L
Linus Torvalds 已提交
939
		       card->battery[0].good ? "OK" : "FAILURE",
940
		       batt_status & BATTERY_2_DISABLED ? "Disabled" : "Enabled",
L
Linus Torvalds 已提交
941 942 943 944 945 946 947 948 949 950 951 952 953 954
		       card->battery[1].good ? "OK" : "FAILURE");

		set_fault_to_battery_status(card);
	}

	pci_read_config_dword(dev, PCI_BASE_ADDRESS_1, &saved_bar);
	data = 0xffffffff;
	pci_write_config_dword(dev, PCI_BASE_ADDRESS_1, data);
	pci_read_config_dword(dev, PCI_BASE_ADDRESS_1, &data);
	pci_write_config_dword(dev, PCI_BASE_ADDRESS_1, saved_bar);
	data &= 0xfffffff0;
	data = ~data;
	data += 1;

955 956
	if (request_irq(dev->irq, mm_interrupt, IRQF_SHARED, DRIVER_NAME,
			card)) {
J
Jeff Garzik 已提交
957 958
		dev_printk(KERN_ERR, &card->dev->dev,
			"Unable to allocate IRQ\n");
L
Linus Torvalds 已提交
959 960 961 962
		ret = -ENODEV;
		goto failed_req_irq;
	}

J
Jeff Garzik 已提交
963
	dev_printk(KERN_INFO, &card->dev->dev,
J
Jeff Garzik 已提交
964
		"Window size %d bytes, IRQ %d\n", data, dev->irq);
L
Linus Torvalds 已提交
965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981

	pci_set_drvdata(dev, card);

	if (pci_write_cmd != 0x0F) 	/* If not Memory Write & Invalidate */
		pci_write_cmd = 0x07;	/* then Memory Write command */

	if (pci_write_cmd & 0x08) { /* use Memory Write and Invalidate */
		unsigned short cfg_command;
		pci_read_config_word(dev, PCI_COMMAND, &cfg_command);
		cfg_command |= 0x10; /* Memory Write & Invalidate Enable */
		pci_write_config_word(dev, PCI_COMMAND, cfg_command);
	}
	pci_cmds = (pci_read_cmd << 28) | (pci_write_cmd << 24);

	num_cards++;

	if (!get_userbit(card, MEMORY_INITIALIZED)) {
J
Jeff Garzik 已提交
982
		dev_printk(KERN_INFO, &card->dev->dev,
983
		  "memory NOT initialized. Consider over-writing whole device.\n");
L
Linus Torvalds 已提交
984 985
		card->init_size = 0;
	} else {
J
Jeff Garzik 已提交
986 987
		dev_printk(KERN_INFO, &card->dev->dev,
			"memory already initialized\n");
L
Linus Torvalds 已提交
988 989 990 991 992 993 994 995 996 997 998
		card->init_size = card->mm_size;
	}

	/* Enable ECC */
	writeb(EDC_STORE_CORRECT, card->csr_remap + MEMCTRLCMD_ERRCTRL);

	return 0;

 failed_req_irq:
 failed_alloc:
	if (card->mm_pages[0].desc)
999 1000 1001
		dma_free_coherent(&card->dev->dev, PAGE_SIZE * 2,
				  card->mm_pages[0].desc,
				  card->mm_pages[0].page_dma);
L
Linus Torvalds 已提交
1002
	if (card->mm_pages[1].desc)
1003 1004 1005
		dma_free_coherent(&card->dev->dev, PAGE_SIZE * 2,
				  card->mm_pages[1].desc,
				  card->mm_pages[1].page_dma);
L
Linus Torvalds 已提交
1006 1007 1008
 failed_magic:
	iounmap(card->csr_remap);
 failed_remap_csr:
J
Jeff Garzik 已提交
1009
	pci_release_regions(dev);
L
Linus Torvalds 已提交
1010 1011 1012 1013
 failed_req_csr:

	return ret;
}
1014

L
Linus Torvalds 已提交
1015 1016 1017 1018 1019
static void mm_pci_remove(struct pci_dev *dev)
{
	struct cardinfo *card = pci_get_drvdata(dev);

	tasklet_kill(&card->tasklet);
J
Jeff Garzik 已提交
1020
	free_irq(dev->irq, card);
L
Linus Torvalds 已提交
1021 1022 1023
	iounmap(card->csr_remap);

	if (card->mm_pages[0].desc)
1024
		dma_free_coherent(&card->dev->dev, PAGE_SIZE * 2,
L
Linus Torvalds 已提交
1025 1026 1027
				    card->mm_pages[0].desc,
				    card->mm_pages[0].page_dma);
	if (card->mm_pages[1].desc)
1028
		dma_free_coherent(&card->dev->dev, PAGE_SIZE * 2,
L
Linus Torvalds 已提交
1029 1030
				    card->mm_pages[1].desc,
				    card->mm_pages[1].page_dma);
1031
	blk_cleanup_queue(card->queue);
J
Jeff Garzik 已提交
1032 1033 1034

	pci_release_regions(dev);
	pci_disable_device(dev);
L
Linus Torvalds 已提交
1035 1036
}

1037
static const struct pci_device_id mm_pci_ids[] = {
1038 1039 1040
    {PCI_DEVICE(PCI_VENDOR_ID_MICRO_MEMORY, PCI_DEVICE_ID_MICRO_MEMORY_5415CN)},
    {PCI_DEVICE(PCI_VENDOR_ID_MICRO_MEMORY, PCI_DEVICE_ID_MICRO_MEMORY_5425CN)},
    {PCI_DEVICE(PCI_VENDOR_ID_MICRO_MEMORY, PCI_DEVICE_ID_MICRO_MEMORY_6155)},
1041
    {
L
Linus Torvalds 已提交
1042 1043
	.vendor	=	0x8086,
	.device	=	0xB555,
1044 1045 1046 1047
	.subvendor =	0x1332,
	.subdevice =	0x5460,
	.class =	0x050000,
	.class_mask =	0,
1048
    }, { /* end: all zeroes */ }
L
Linus Torvalds 已提交
1049 1050 1051 1052 1053
};

MODULE_DEVICE_TABLE(pci, mm_pci_ids);

static struct pci_driver mm_pci_driver = {
J
Jeff Garzik 已提交
1054 1055 1056 1057
	.name		= DRIVER_NAME,
	.id_table	= mm_pci_ids,
	.probe		= mm_pci_probe,
	.remove		= mm_pci_remove,
L
Linus Torvalds 已提交
1058
};
J
Jeff Garzik 已提交
1059

L
Linus Torvalds 已提交
1060 1061 1062 1063 1064
static int __init mm_init(void)
{
	int retval, i;
	int err;

1065
	retval = pci_register_driver(&mm_pci_driver);
L
Linus Torvalds 已提交
1066 1067 1068
	if (retval)
		return -ENOMEM;

1069
	err = major_nr = register_blkdev(0, DRIVER_NAME);
1070 1071
	if (err < 0) {
		pci_unregister_driver(&mm_pci_driver);
L
Linus Torvalds 已提交
1072
		return -EIO;
1073
	}
L
Linus Torvalds 已提交
1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094

	for (i = 0; i < num_cards; i++) {
		mm_gendisk[i] = alloc_disk(1 << MM_SHIFT);
		if (!mm_gendisk[i])
			goto out;
	}

	for (i = 0; i < num_cards; i++) {
		struct gendisk *disk = mm_gendisk[i];
		sprintf(disk->disk_name, "umem%c", 'a'+i);
		spin_lock_init(&cards[i].lock);
		disk->major = major_nr;
		disk->first_minor  = i << MM_SHIFT;
		disk->fops = &mm_fops;
		disk->private_data = &cards[i];
		disk->queue = cards[i].queue;
		set_capacity(disk, cards[i].mm_size << 1);
		add_disk(disk);
	}

	init_battery_timer();
J
Jeff Garzik 已提交
1095
	printk(KERN_INFO "MM: desc_per_page = %ld\n", DESC_PER_PAGE);
L
Linus Torvalds 已提交
1096 1097 1098 1099
/* printk("mm_init: Done. 10-19-01 9:00\n"); */
	return 0;

out:
1100
	pci_unregister_driver(&mm_pci_driver);
1101
	unregister_blkdev(major_nr, DRIVER_NAME);
L
Linus Torvalds 已提交
1102 1103 1104 1105
	while (i--)
		put_disk(mm_gendisk[i]);
	return -ENOMEM;
}
1106

L
Linus Torvalds 已提交
1107 1108 1109 1110 1111 1112
static void __exit mm_cleanup(void)
{
	int i;

	del_battery_timer();

1113
	for (i = 0; i < num_cards ; i++) {
L
Linus Torvalds 已提交
1114 1115 1116 1117 1118 1119
		del_gendisk(mm_gendisk[i]);
		put_disk(mm_gendisk[i]);
	}

	pci_unregister_driver(&mm_pci_driver);

1120
	unregister_blkdev(major_nr, DRIVER_NAME);
L
Linus Torvalds 已提交
1121 1122 1123 1124 1125 1126 1127 1128
}

module_init(mm_init);
module_exit(mm_cleanup);

MODULE_AUTHOR(DRIVER_AUTHOR);
MODULE_DESCRIPTION(DRIVER_DESC);
MODULE_LICENSE("GPL");