⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 dasd_eckd.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 4 页
字号:
	if (recid < 2 * blk_per_trk)		return 1;	return 0;}/* * Returns the record size for the special blocks of the cdl format. * Only returns something useful if dasd_eckd_cdl_special is true * for the recid. */static inline intdasd_eckd_cdl_reclen(int recid){	if (recid < 3)		return sizes_trk0[recid];	return LABEL_SIZE;}static intdasd_eckd_read_conf(struct dasd_device *device){	void *conf_data;	int conf_len, conf_data_saved;	int rc;	__u8 lpm;	struct dasd_eckd_private *private;	struct dasd_eckd_path *path_data;	private = (struct dasd_eckd_private *) device->private;	path_data = (struct dasd_eckd_path *) &private->path_data;	path_data->opm = ccw_device_get_path_mask(device->cdev);	lpm = 0x80;	conf_data_saved = 0;	/* get configuration data per operational path */	for (lpm = 0x80; lpm; lpm>>= 1) {		if (lpm & path_data->opm){			rc = read_conf_data_lpm(device->cdev, &conf_data,						&conf_len, lpm);			if (rc && rc != -EOPNOTSUPP) {	/* -EOPNOTSUPP is ok */				MESSAGE(KERN_WARNING,					"Read configuration data returned "					"error %d", rc);				return rc;			}			if (conf_data == NULL) {				MESSAGE(KERN_WARNING, "%s", "No configuration "					"data retrieved");				continue;	/* no errror */			}			if (conf_len != sizeof (struct dasd_eckd_confdata)) {				MESSAGE(KERN_WARNING,					"sizes of configuration data mismatch"					"%d (read) vs %ld (expected)",					conf_len,					sizeof (struct dasd_eckd_confdata));				kfree(conf_data);				continue;	/* no errror */			}			/* save first valid configuration data */			if (!conf_data_saved){				memcpy(&private->conf_data, conf_data,				       sizeof (struct dasd_eckd_confdata));				conf_data_saved++;			}			switch (((char *)conf_data)[242] & 0x07){			case 0x02:				path_data->npm |= lpm;				break;			case 0x03:				path_data->ppm |= lpm;				break;			}			kfree(conf_data);		}	}	return 0;}static intdasd_eckd_check_characteristics(struct dasd_device *device){	struct dasd_eckd_private *private;	void *rdc_data;	int rc;	private = (struct dasd_eckd_private *) device->private;	if (private == NULL) {		private = kmalloc(sizeof(struct dasd_eckd_private),				  GFP_KERNEL | GFP_DMA);		if (private == NULL) {			DEV_MESSAGE(KERN_WARNING, device, "%s",				    "memory allocation failed for private "				    "data");			return -ENOMEM;		}		memset(private, 0, sizeof(struct dasd_eckd_private));		device->private = (void *) private;	}	/* Invalidate status of initial analysis. */	private->init_cqr_status = -1;	/* Set default cache operations. */	private->attrib.operation = DASD_NORMAL_CACHE;	private->attrib.nr_cyl = 0;	/* Read Device Characteristics */	rdc_data = (void *) &(private->rdc_data);	rc = read_dev_chars(device->cdev, &rdc_data, 64);	if (rc) {		DEV_MESSAGE(KERN_WARNING, device,			    "Read device characteristics returned error %d",			    rc);		return rc;	}	DEV_MESSAGE(KERN_INFO, device,		    "%04X/%02X(CU:%04X/%02X) Cyl:%d Head:%d Sec:%d",		    private->rdc_data.dev_type,		    private->rdc_data.dev_model,		    private->rdc_data.cu_type,		    private->rdc_data.cu_model.model,		    private->rdc_data.no_cyl,		    private->rdc_data.trk_per_cyl,		    private->rdc_data.sec_per_trk);	/* Read Configuration Data */	rc = dasd_eckd_read_conf (device);	return rc;}static struct dasd_ccw_req *dasd_eckd_analysis_ccw(struct dasd_device *device){	struct dasd_eckd_private *private;	struct eckd_count *count_data;	struct LO_eckd_data *LO_data;	struct dasd_ccw_req *cqr;	struct ccw1 *ccw;	int cplength, datasize;	int i;	private = (struct dasd_eckd_private *) device->private;	cplength = 8;	datasize = sizeof(struct DE_eckd_data) + 2*sizeof(struct LO_eckd_data);	cqr = dasd_smalloc_request(dasd_eckd_discipline.name,				   cplength, datasize, device);	if (IS_ERR(cqr))		return cqr;	ccw = cqr->cpaddr;	/* Define extent for the first 3 tracks. */	define_extent(ccw++, cqr->data, 0, 2,		      DASD_ECKD_CCW_READ_COUNT, device);	LO_data = cqr->data + sizeof (struct DE_eckd_data);	/* Locate record for the first 4 records on track 0. */	ccw[-1].flags |= CCW_FLAG_CC;	locate_record(ccw++, LO_data++, 0, 0, 4,		      DASD_ECKD_CCW_READ_COUNT, device, 0);	count_data = private->count_area;	for (i = 0; i < 4; i++) {		ccw[-1].flags |= CCW_FLAG_CC;		ccw->cmd_code = DASD_ECKD_CCW_READ_COUNT;		ccw->flags = 0;		ccw->count = 8;		ccw->cda = (__u32)(addr_t) count_data;		ccw++;		count_data++;	}	/* Locate record for the first record on track 2. */	ccw[-1].flags |= CCW_FLAG_CC;	locate_record(ccw++, LO_data++, 2, 0, 1,		      DASD_ECKD_CCW_READ_COUNT, device, 0);	/* Read count ccw. */	ccw[-1].flags |= CCW_FLAG_CC;	ccw->cmd_code = DASD_ECKD_CCW_READ_COUNT;	ccw->flags = 0;	ccw->count = 8;	ccw->cda = (__u32)(addr_t) count_data;	cqr->device = device;	cqr->retries = 0;	cqr->buildclk = get_clock();	cqr->status = DASD_CQR_FILLED;	return cqr;}/* * This is the callback function for the init_analysis cqr. It saves * the status of the initial analysis ccw before it frees it and kicks * the device to continue the startup sequence. This will call * dasd_eckd_do_analysis again (if the devices has not been marked * for deletion in the meantime). */static voiddasd_eckd_analysis_callback(struct dasd_ccw_req *init_cqr, void *data){	struct dasd_eckd_private *private;	struct dasd_device *device;	device = init_cqr->device;	private = (struct dasd_eckd_private *) device->private;	private->init_cqr_status = init_cqr->status;	dasd_sfree_request(init_cqr, device);	dasd_kick_device(device);}static intdasd_eckd_start_analysis(struct dasd_device *device){	struct dasd_eckd_private *private;	struct dasd_ccw_req *init_cqr;	private = (struct dasd_eckd_private *) device->private;	init_cqr = dasd_eckd_analysis_ccw(device);	if (IS_ERR(init_cqr))		return PTR_ERR(init_cqr);	init_cqr->callback = dasd_eckd_analysis_callback;	init_cqr->callback_data = NULL;	init_cqr->expires = 5*HZ;	dasd_add_request_head(init_cqr);	return -EAGAIN;}static intdasd_eckd_end_analysis(struct dasd_device *device){	struct dasd_eckd_private *private;	struct eckd_count *count_area;	unsigned int sb, blk_per_trk;	int status, i;	private = (struct dasd_eckd_private *) device->private;	status = private->init_cqr_status;	private->init_cqr_status = -1;	if (status != DASD_CQR_DONE) {		DEV_MESSAGE(KERN_WARNING, device, "%s",			    "volume analysis returned unformatted disk");		return -EMEDIUMTYPE;	}	private->uses_cdl = 1;	/* Calculate number of blocks/records per track. */	blk_per_trk = recs_per_track(&private->rdc_data, 0, device->bp_block);	/* Check Track 0 for Compatible Disk Layout */	count_area = NULL;	for (i = 0; i < 3; i++) {		if (private->count_area[i].kl != 4 ||		    private->count_area[i].dl != dasd_eckd_cdl_reclen(i) - 4) {			private->uses_cdl = 0;			break;		}	}	if (i == 3)		count_area = &private->count_area[4];	if (private->uses_cdl == 0) {		for (i = 0; i < 5; i++) {			if ((private->count_area[i].kl != 0) ||			    (private->count_area[i].dl !=			     private->count_area[0].dl))				break;		}		if (i == 5)			count_area = &private->count_area[0];	} else {		if (private->count_area[3].record == 1)			DEV_MESSAGE(KERN_WARNING, device, "%s",				    "Trk 0: no records after VTOC!");	}	if (count_area != NULL && count_area->kl == 0) {		/* we found notthing violating our disk layout */		if (dasd_check_blocksize(count_area->dl) == 0)			device->bp_block = count_area->dl;	}	if (device->bp_block == 0) {		DEV_MESSAGE(KERN_WARNING, device, "%s",			    "Volume has incompatible disk layout");		return -EMEDIUMTYPE;	}	device->s2b_shift = 0;	/* bits to shift 512 to get a block */	for (sb = 512; sb < device->bp_block; sb = sb << 1)		device->s2b_shift++;	blk_per_trk = recs_per_track(&private->rdc_data, 0, device->bp_block);	device->blocks = (private->rdc_data.no_cyl *			  private->rdc_data.trk_per_cyl *			  blk_per_trk);	DEV_MESSAGE(KERN_INFO, device,		    "(%dkB blks): %dkB at %dkB/trk %s",		    (device->bp_block >> 10),		    ((private->rdc_data.no_cyl *		      private->rdc_data.trk_per_cyl *		      blk_per_trk * (device->bp_block >> 9)) >> 1),		    ((blk_per_trk * device->bp_block) >> 10), 		    private->uses_cdl ?		    "compatible disk layout" : "linux disk layout");	return 0;}static intdasd_eckd_do_analysis(struct dasd_device *device){	struct dasd_eckd_private *private;	private = (struct dasd_eckd_private *) device->private;	if (private->init_cqr_status < 0)		return dasd_eckd_start_analysis(device);	else		return dasd_eckd_end_analysis(device);}static intdasd_eckd_fill_geometry(struct dasd_device *device, struct hd_geometry *geo){	struct dasd_eckd_private *private;	private = (struct dasd_eckd_private *) device->private;	if (dasd_check_blocksize(device->bp_block) == 0) {		geo->sectors = recs_per_track(&private->rdc_data,					      0, device->bp_block);	}	geo->cylinders = private->rdc_data.no_cyl;	geo->heads = private->rdc_data.trk_per_cyl;	return 0;}static struct dasd_ccw_req *dasd_eckd_format_device(struct dasd_device * device,			struct format_data_t * fdata){	struct dasd_eckd_private *private;	struct dasd_ccw_req *fcp;	struct eckd_count *ect;	struct ccw1 *ccw;	void *data;	int rpt, cyl, head;	int cplength, datasize;	int i;	private = (struct dasd_eckd_private *) device->private;	rpt = recs_per_track(&private->rdc_data, 0, fdata->blksize);	cyl = fdata->start_unit / private->rdc_data.trk_per_cyl;	head = fdata->start_unit % private->rdc_data.trk_per_cyl;	/* Sanity checks. */	if (fdata->start_unit >=	    (private->rdc_data.no_cyl * private->rdc_data.trk_per_cyl)) {		DEV_MESSAGE(KERN_INFO, device, "Track no %d too big!",			    fdata->start_unit);		return ERR_PTR(-EINVAL);	}	if (fdata->start_unit > fdata->stop_unit) {		DEV_MESSAGE(KERN_INFO, device, "Track %d reached! ending.",			    fdata->start_unit);		return ERR_PTR(-EINVAL);	}	if (dasd_check_blocksize(fdata->blksize) != 0) {		DEV_MESSAGE(KERN_WARNING, device,			    "Invalid blocksize %d...terminating!",			    fdata->blksize);		return ERR_PTR(-EINVAL);	}	/*	 * fdata->intensity is a bit string that tells us what to do:	 *   Bit 0: write record zero	 *   Bit 1: write home address, currently not supported	 *   Bit 2: invalidate tracks	 *   Bit 3: use OS/390 compatible disk layout (cdl)	 * Only some bit combinations do make sense.	 */	switch (fdata->intensity) {	case 0x00:	/* Normal format */	case 0x08:	/* Normal format, use cdl. */		cplength = 2 + rpt;		datasize = sizeof(struct DE_eckd_data) +			sizeof(struct LO_eckd_data) +			rpt * sizeof(struct eckd_count);		break;	case 0x01:	/* Write record zero and format track. */	case 0x09:	/* Write record zero and format track, use cdl. */		cplength = 3 + rpt;		datasize = sizeof(struct DE_eckd_data) +			sizeof(struct LO_eckd_data) +			sizeof(struct eckd_count) +			rpt * sizeof(struct eckd_count);		break;	case 0x04:	/* Invalidate track. */	case 0x0c:	/* Invalidate track, use cdl. */		cplength = 3;		datasize = sizeof(struct DE_eckd_data) +			sizeof(struct LO_eckd_data) +			sizeof(struct eckd_count);		break;	default:		DEV_MESSAGE(KERN_WARNING, device, "Invalid flags 0x%x.",			    fdata->intensity);		return ERR_PTR(-EINVAL);	}	/* Allocate the format ccw request. */	fcp = dasd_smalloc_request(dasd_eckd_discipline.name,				   cplength, datasize, device);	if (IS_ERR(fcp))		return fcp;	data = fcp->data;	ccw = fcp->cpaddr;	switch (fdata->intensity & ~0x08) {	case 0x00: /* Normal format. */		define_extent(ccw++, (struct DE_eckd_data *) data,			      fdata->start_unit, fdata->start_unit,			      DASD_ECKD_CCW_WRITE_CKD, device);		data += sizeof(struct DE_eckd_data);		ccw[-1].flags |= CCW_FLAG_CC;		locate_record(ccw++, (struct LO_eckd_data *) data,			      fdata->start_unit, 0, rpt,			      DASD_ECKD_CCW_WRITE_CKD, device,			      fdata->blksize);		data += sizeof(struct LO_eckd_data);		break;	case 0x01: /* Write record zero + format track. */		define_extent(ccw++, (struct DE_eckd_data *) data,			      fdata->start_unit, fdata->start_unit,			      DASD_ECKD_CCW_WRITE_RECORD_ZERO,

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -