patch-2.4.2 linux/drivers/s390/block/dasd_fba.c
Next file: linux/drivers/s390/block/dasd_fba.h
Previous file: linux/drivers/s390/block/dasd_erp.h
Back to the patch index
Back to the overall index
- Lines: 440
- Date:
Tue Feb 13 14:13:44 2001
- Orig file:
v2.4.1/linux/drivers/s390/block/dasd_fba.c
- Orig date:
Wed Dec 31 16:00:00 1969
diff -u --recursive --new-file v2.4.1/linux/drivers/s390/block/dasd_fba.c linux/drivers/s390/block/dasd_fba.c
@@ -0,0 +1,439 @@
+
+/*
+ * File...........: linux/drivers/s390/block/dasd_fba.c
+ * Author(s)......: Holger Smolinski <Holger.Smolinski@de.ibm.com>
+ * Bugreports.to..: <Linux390@de.ibm.com>
+ * (C) IBM Corporation, IBM Deutschland Entwicklung GmbH, 1999,2000
+ */
+
+#include <linux/config.h>
+#include <linux/stddef.h>
+#include <linux/kernel.h>
+#include <asm/debug.h>
+
+#include <linux/malloc.h>
+#include <linux/hdreg.h> /* HDIO_GETGEO */
+#include <linux/blk.h>
+#include <asm/ccwcache.h>
+#include <asm/idals.h>
+#include <asm/dasd.h>
+
+#include <asm/ebcdic.h>
+#include <asm/io.h>
+#include <asm/irq.h>
+#include <asm/s390dyn.h>
+
+#include "dasd_fba.h"
+
+#ifdef PRINTK_HEADER
+#undef PRINTK_HEADER
+#endif /* PRINTK_HEADER */
+#define PRINTK_HEADER DASD_NAME"(fba):"
+
+#define DASD_FBA_CCW_WRITE 0x41
+#define DASD_FBA_CCW_READ 0x42
+#define DASD_FBA_CCW_LOCATE 0x43
+#define DASD_FBA_CCW_DEFINE_EXTENT 0x63
+
+dasd_discipline_t dasd_fba_discipline;
+
+typedef struct
+dasd_fba_private_t {
+ dasd_fba_characteristics_t rdc_data;
+} dasd_fba_private_t;
+
+#ifdef CONFIG_DASD_DYNAMIC
+static
+devreg_t dasd_fba_known_devices[] =
+{
+ {
+ ci:
+ {hc:
+ {ctype:0x6310, dtype:0x9336}},
+ flag:DEVREG_MATCH_CU_TYPE | DEVREG_MATCH_DEV_TYPE,
+ oper_func:dasd_oper_handler
+ },
+ {
+ ci:
+ {hc:
+ {ctype:0x3880, dtype:0x3370}},
+ flag:DEVREG_MATCH_CU_TYPE | DEVREG_MATCH_DEV_TYPE,
+ oper_func:dasd_oper_handler
+ }
+};
+#endif
+static inline void
+define_extent (ccw1_t * ccw, DE_fba_data_t * DE_data, int rw,
+ int blksize, int beg, int nr)
+{
+ memset (DE_data, 0, sizeof (DE_fba_data_t));
+ ccw->cmd_code = DASD_FBA_CCW_DEFINE_EXTENT;
+ ccw->count = 16;
+ set_normalized_cda (ccw, __pa (DE_data));
+ if (rw == WRITE)
+ (DE_data->mask).perm = 0x0;
+ else if (rw == READ)
+ (DE_data->mask).perm = 0x1;
+ else
+ DE_data->mask.perm = 0x2;
+ DE_data->blk_size = blksize;
+ DE_data->ext_loc = beg;
+ DE_data->ext_end = nr - 1;
+}
+
+static inline void
+locate_record (ccw1_t * ccw, LO_fba_data_t * LO_data, int rw, int block_nr,
+ int block_ct)
+{
+ memset (LO_data, 0, sizeof (LO_fba_data_t));
+ ccw->cmd_code = DASD_FBA_CCW_LOCATE;
+ ccw->count = 8;
+ set_normalized_cda (ccw, __pa (LO_data));
+ if (rw == WRITE)
+ LO_data->operation.cmd = 0x5;
+ else if (rw == READ)
+ LO_data->operation.cmd = 0x6;
+ else
+ LO_data->operation.cmd = 0x8;
+ LO_data->blk_nr = block_nr;
+ LO_data->blk_ct = block_ct;
+}
+
+static int
+dasd_fba_id_check (s390_dev_info_t * info)
+{
+ if (info->sid_data.cu_type == 0x3880)
+ if (info->sid_data.dev_type == 0x3370)
+ return 0;
+ if (info->sid_data.cu_type == 0x6310)
+ if (info->sid_data.dev_type == 0x9336)
+ return 0;
+ return -ENODEV;
+}
+
+static int
+dasd_fba_check_characteristics (struct dasd_device_t *device)
+{
+ int rc = -ENODEV;
+ void *rdc_data;
+ dasd_fba_private_t *private;
+
+ if (device == NULL) {
+ printk (KERN_WARNING PRINTK_HEADER
+ "Null device pointer passed to characteristics checker\n");
+ return -ENODEV;
+ }
+ if (device->private == NULL) {
+ device->private = kmalloc (sizeof (dasd_fba_private_t), GFP_KERNEL);
+ if (device->private == NULL) {
+ printk (KERN_WARNING PRINTK_HEADER
+ "memory allocation failed for private data\n");
+ return -ENOMEM;
+ }
+ }
+ private = (dasd_fba_private_t *) device->private;
+ rdc_data = (void *) &(private->rdc_data);
+ rc = read_dev_chars (device->devinfo.irq, &rdc_data, 32);
+ if (rc) {
+ printk (KERN_WARNING PRINTK_HEADER
+ "Read device characteristics returned error %d\n", rc);
+ kfree(private);
+ device->private=NULL;
+ return rc;
+ }
+ printk (KERN_INFO PRINTK_HEADER
+ "%04X on sch %d: %04X/%02X(CU:%04X/%02X) %dMB at(%d B/blk)\n",
+ device->devinfo.devno, device->devinfo.irq,
+ device->devinfo.sid_data.dev_type, device->devinfo.sid_data.dev_model,
+ device->devinfo.sid_data.cu_type, device->devinfo.sid_data.cu_model,
+ (private->rdc_data.blk_bdsa *
+ (private->rdc_data.blk_size >> 9)) >> 11,
+ private->rdc_data.blk_size);
+ return 0;
+}
+
+static int
+dasd_fba_do_analysis (struct dasd_device_t *device)
+{
+ int rc = 0;
+ int sb;
+ dasd_fba_private_t *private = (dasd_fba_private_t *) device->private;
+ int bs = private->rdc_data.blk_size;
+
+ memset (&(device->sizes), 0, sizeof (dasd_sizes_t));
+ switch (bs) {
+ case 512:
+ case 1024:
+ case 2048:
+ case 4096:
+ device->sizes.bp_block = bs;
+ break;
+ default:
+ printk (KERN_INFO PRINTK_HEADER
+ "/dev/%s (%04X): unknown blocksize %d\n",
+ device->name, device->devinfo.devno, bs);
+ return -EMEDIUMTYPE;
+ }
+ device->sizes.s2b_shift = 0; /* bits to shift 512 to get a block */
+ for (sb = 512; sb < bs; sb = sb << 1)
+ device->sizes.s2b_shift++;
+
+ device->sizes.blocks = (private->rdc_data.blk_bdsa);
+
+ return rc;
+}
+
+static int
+dasd_fba_fill_geometry (struct dasd_device_t *device, struct hd_geometry *geo)
+{
+ int rc = 0;
+ unsigned long sectors = device->sizes.blocks << device->sizes.s2b_shift;
+ unsigned long tracks = sectors >> 6;
+ unsigned long cyls = tracks >> 4;
+
+ switch (device->sizes.bp_block) {
+ case 512:
+ case 1024:
+ case 2048:
+ case 4096:
+ break;
+ default:
+ return -EINVAL;
+ }
+ geo->cylinders = cyls;
+ geo->heads = 16;
+ geo->sectors = 128 >> device->sizes.s2b_shift;
+ geo->start = 1;
+ return rc;
+}
+
+static dasd_era_t
+dasd_fba_examine_error (ccw_req_t * cqr, devstat_t * stat)
+{
+ dasd_device_t *device = (dasd_device_t *) cqr->device;
+ if (stat->cstat == 0x00 &&
+ stat->dstat == (DEV_STAT_CHN_END | DEV_STAT_DEV_END))
+ return dasd_era_none;
+
+ switch (device->devinfo.sid_data.dev_model) {
+ case 0x3370:
+ return dasd_3370_erp_examine (cqr, stat);
+ case 0x9336:
+ return dasd_9336_erp_examine (cqr, stat);
+ default:
+ return dasd_era_recover;
+ }
+}
+
+static dasd_erp_action_fn_t
+dasd_fba_erp_action (ccw_req_t * cqr)
+{
+ return default_erp_action;
+}
+
+static dasd_erp_postaction_fn_t
+dasd_fba_erp_postaction (ccw_req_t * cqr)
+{
+ if (cqr->function == default_erp_action)
+ return default_erp_postaction;
+ printk (KERN_WARNING PRINTK_HEADER
+ "unknown ERP action %p\n",
+ cqr->function);
+ return NULL;
+}
+
+static ccw_req_t *
+dasd_fba_build_cp_from_req (dasd_device_t * device, struct request *req)
+{
+ ccw_req_t *rw_cp = NULL;
+ int rw_cmd;
+ int bhct;
+ long size;
+ ccw1_t *ccw;
+ DE_fba_data_t *DE_data;
+ LO_fba_data_t *LO_data;
+ struct buffer_head *bh;
+ dasd_fba_private_t *private = (dasd_fba_private_t *) device->private;
+ int byt_per_blk = device->sizes.bp_block;
+
+ if (req->cmd == READ) {
+ rw_cmd = DASD_FBA_CCW_READ;
+ } else if (req->cmd == WRITE) {
+ rw_cmd = DASD_FBA_CCW_WRITE;
+ } else {
+ PRINT_ERR ("Unknown command %d\n", req->cmd);
+ return NULL;
+ }
+ /* Build the request */
+ /* count bhs to prevent errors, when bh smaller than block */
+ bhct = 0;
+ for (bh = req->bh; bh; bh = bh->b_reqnext) {
+ if (bh->b_size > byt_per_blk)
+ for (size = 0; size < bh->b_size; size += byt_per_blk)
+ bhct++;
+ else
+ bhct++;
+ }
+
+ rw_cp = dasd_alloc_request (dasd_fba_discipline.name,
+ 2 + bhct,
+ sizeof (DE_fba_data_t) +
+ sizeof (LO_fba_data_t));
+ if (!rw_cp) {
+ return NULL;
+ }
+ DE_data = rw_cp->data;
+ LO_data = rw_cp->data + sizeof (DE_fba_data_t);
+ ccw = rw_cp->cpaddr;
+
+ define_extent (ccw, DE_data, req->cmd, byt_per_blk,
+ req->sector, req->nr_sectors);
+ ccw->flags |= CCW_FLAG_CC;
+ ccw++;
+ locate_record (ccw, LO_data, req->cmd, 0, req->nr_sectors);
+ ccw->flags |= CCW_FLAG_CC;
+
+ for (bh = req->bh; bh;) {
+ if (bh->b_size > byt_per_blk) {
+ for (size = 0; size < bh->b_size; size += byt_per_blk) {
+ ccw++;
+ if (private->rdc_data.mode.bits.data_chain) {
+ ccw->flags |= CCW_FLAG_DC;
+ } else {
+ ccw->flags |= CCW_FLAG_CC;
+ }
+ ccw->cmd_code = rw_cmd;
+ ccw->count = byt_per_blk;
+ set_normalized_cda (ccw, __pa (bh->b_data + size));
+ }
+ bh = bh->b_reqnext;
+ } else { /* group N bhs to fit into byt_per_blk */
+ for (size = 0; bh != NULL && size < byt_per_blk;) {
+ ccw++;
+ if (private->rdc_data.mode.bits.data_chain) {
+ ccw->flags |= CCW_FLAG_DC;
+ } else {
+ PRINT_WARN ("Cannot chain chunks smaller than one block\n");
+ ccw_free_request (rw_cp);
+ return NULL;
+ }
+ ccw->cmd_code = rw_cmd;
+ ccw->count = bh->b_size;
+ set_normalized_cda (ccw, __pa (bh->b_data));
+ size += bh->b_size;
+ bh = bh->b_reqnext;
+ }
+ ccw->flags &= ~CCW_FLAG_DC;
+ ccw->flags |= CCW_FLAG_CC;
+ if (size != byt_per_blk) {
+ PRINT_WARN ("Cannot fulfill request smaller than block\n");
+ ccw_free_request (rw_cp);
+ return NULL;
+ }
+ }
+ }
+ ccw->flags &= ~(CCW_FLAG_DC | CCW_FLAG_CC);
+
+ rw_cp->device = device;
+ rw_cp->expires = 5 * TOD_MIN; /* 5 minutes */
+ rw_cp->req = req;
+ check_then_set (&rw_cp->status, CQR_STATUS_EMPTY, CQR_STATUS_FILLED);
+ return rw_cp;
+}
+
+static char *
+dasd_fba_dump_sense (struct dasd_device_t *device, ccw_req_t * req)
+{
+ char *page = (char *) get_free_page (GFP_KERNEL);
+ int len;
+ if (page == NULL) {
+ return NULL;
+ }
+ len = sprintf (page, KERN_WARNING PRINTK_HEADER
+ "device %04X on irq %d: I/O status report:\n",
+ device->devinfo.devno, device->devinfo.irq);
+
+ return page;
+}
+
+dasd_discipline_t dasd_fba_discipline =
+{
+ name:"FBA ",
+ ebcname:"FBA ",
+ id_check:dasd_fba_id_check,
+ check_characteristics:dasd_fba_check_characteristics,
+ do_analysis:dasd_fba_do_analysis,
+ fill_geometry:dasd_fba_fill_geometry,
+ start_IO:dasd_start_IO,
+ examine_error:dasd_fba_examine_error,
+ erp_action:dasd_fba_erp_action,
+ erp_postaction:dasd_fba_erp_postaction,
+ build_cp_from_req:dasd_fba_build_cp_from_req,
+ dump_sense:dasd_fba_dump_sense,
+ int_handler:dasd_int_handler
+};
+
+
+int
+dasd_fba_init (void)
+{
+ int rc = 0;
+ printk (KERN_INFO PRINTK_HEADER
+ "%s discipline initializing\n", dasd_fba_discipline.name);
+ ASCEBC (dasd_fba_discipline.ebcname, 4);
+ dasd_discipline_enq (&dasd_fba_discipline);
+#ifdef CONFIG_DASD_DYNAMIC
+ {
+ int i;
+ for (i = 0; i < sizeof (dasd_fba_known_devices) / sizeof (devreg_t); i++) {
+ printk (KERN_INFO PRINTK_HEADER
+ "We are interested in: CU %04X/%02x DEV: %04X/%02Xi\n",
+ dasd_fba_known_devices[i].ci.hc.ctype,
+ dasd_fba_known_devices[i].ci.hc.cmode,
+ dasd_fba_known_devices[i].ci.hc.dtype,
+ dasd_fba_known_devices[i].ci.hc.dmode);
+ s390_device_register (&dasd_fba_known_devices[i]);
+ }
+ }
+#endif /* CONFIG_DASD_DYNAMIC */
+ return rc;
+}
+
+void
+dasd_fba_cleanup( void ) {
+ printk ( KERN_INFO PRINTK_HEADER
+ "%s discipline cleaning up\n", dasd_fba_discipline.name);
+#ifdef CONFIG_DASD_DYNAMIC
+ {
+ int i;
+ for ( i=0; i<sizeof(dasd_fba_known_devices)/sizeof(devreg_t); i++) {
+ printk (KERN_INFO PRINTK_HEADER
+ "We are interested in: CU %04X/%02x DEV: %04X/%02Xi\n",
+ dasd_fba_known_devices[i].ci.hc.ctype,
+ dasd_fba_known_devices[i].ci.hc.cmode,
+ dasd_fba_known_devices[i].ci.hc.dtype,
+ dasd_fba_known_devices[i].ci.hc.dmode);
+ s390_device_register(&dasd_fba_known_devices[i]);
+ }
+ }
+#endif /* CONFIG_DASD_DYNAMIC */
+ dasd_discipline_deq(&dasd_fba_discipline);
+}
+/*
+ * Overrides for Emacs so that we follow Linus's tabbing style.
+ * Emacs will notice this stuff at the end of the file and automatically
+ * adjust the settings for this buffer only. This must remain at the end
+ * of the file.
+ * ---------------------------------------------------------------------------
+ * Local variables:
+ * c-indent-level: 4
+ * c-brace-imaginary-offset: 0
+ * c-brace-offset: -4
+ * c-argdecl-indent: 4
+ * c-label-offset: -4
+ * c-continued-statement-offset: 4
+ * c-continued-brace-offset: 0
+ * indent-tabs-mode: nil
+ * tab-width: 8
+ * End:
+ */
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)