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

📄 scsi-disk.c

📁 xen虚拟机源代码安装包
💻 C
📖 第 1 页 / 共 2 页
字号:
        DPRINTF("Inquiry (len %d)\n", len);        if (buf[1] & 0x2) {            /* Command support data - optional, not implemented */            BADF("optional INQUIRY command support request not implemented\n");            goto fail;        }        else if (buf[1] & 0x1) {            /* Vital product data */            uint8_t page_code = buf[2];            if (len < 4) {                BADF("Error: Inquiry (EVPD[%02X]) buffer size %d is "                     "less than 4\n", page_code, len);                goto fail;            }            switch (page_code) {                case 0x00:                    {                        /* Supported page codes, mandatory */                        DPRINTF("Inquiry EVPD[Supported pages] "                                "buffer size %d\n", len);                        r->buf_len = 0;                        if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) {                            outbuf[r->buf_len++] = 5;                        } else {                            outbuf[r->buf_len++] = 0;                        }                        outbuf[r->buf_len++] = 0x00; // this page                        outbuf[r->buf_len++] = 0x00;                        outbuf[r->buf_len++] = 3;    // number of pages                        outbuf[r->buf_len++] = 0x00; // list of supported pages (this page)                        outbuf[r->buf_len++] = 0x80; // unit serial number                        outbuf[r->buf_len++] = 0x83; // device identification                    }                    break;                case 0x80:                    {                        /* Device serial number, optional */                        if (len < 4) {                            BADF("Error: EVPD[Serial number] Inquiry buffer "                                 "size %d too small, %d needed\n", len, 4);                            goto fail;                        }                        DPRINTF("Inquiry EVPD[Serial number] buffer size %d\n", len);                        r->buf_len = 0;                        /* Supported page codes */                        if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) {                            outbuf[r->buf_len++] = 5;                        } else {                            outbuf[r->buf_len++] = 0;                        }                        outbuf[r->buf_len++] = 0x80; // this page                        outbuf[r->buf_len++] = 0x00;                        outbuf[r->buf_len++] = 0x01; // 1 byte data follow                        outbuf[r->buf_len++] = '0';  // 1 byte data follow                     }                    break;                case 0x83:                    {                        /* Device identification page, mandatory */                        int max_len = 255 - 8;                        int id_len = strlen(bdrv_get_device_name(s->bdrv));                        if (id_len > max_len)                            id_len = max_len;                        DPRINTF("Inquiry EVPD[Device identification] "                                "buffer size %d\n", len);                        r->buf_len = 0;                        if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) {                            outbuf[r->buf_len++] = 5;                        } else {                            outbuf[r->buf_len++] = 0;                        }                        outbuf[r->buf_len++] = 0x83; // this page                        outbuf[r->buf_len++] = 0x00;                        outbuf[r->buf_len++] = 3 + id_len;                        outbuf[r->buf_len++] = 0x2; // ASCII                        outbuf[r->buf_len++] = 0;   // not officially assigned                        outbuf[r->buf_len++] = 0;   // reserved                        outbuf[r->buf_len++] = id_len; // length of data following                        memcpy(&outbuf[r->buf_len],                               bdrv_get_device_name(s->bdrv), id_len);                        r->buf_len += id_len;                    }                    break;                default:                    BADF("Error: unsupported Inquiry (EVPD[%02X]) "                         "buffer size %d\n", page_code, len);                    goto fail;            }            /* done with EVPD */            break;        }        else {            /* Standard INQUIRY data */            if (buf[2] != 0) {                BADF("Error: Inquiry (STANDARD) page or code "                     "is non-zero [%02X]\n", buf[2]);                goto fail;            }            /* PAGE CODE == 0 */            if (len < 5) {                BADF("Error: Inquiry (STANDARD) buffer size %d "                     "is less than 5\n", len);                goto fail;            }            if (len < 36) {                BADF("Error: Inquiry (STANDARD) buffer size %d "                     "is less than 36 (TODO: only 5 required)\n", len);            }        }	memset(outbuf, 0, 36);	if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) {	    outbuf[0] = 5;            outbuf[1] = 0x80;	    memcpy(&outbuf[16], "QEMU CD-ROM    ", 16);	} else {	    outbuf[0] = 0;	    memcpy(&outbuf[16], "QEMU HARDDISK  ", 16);	}	memcpy(&outbuf[8], "QEMU   ", 8);        memcpy(&outbuf[32], QEMU_VERSION, 4);        /* Identify device as SCSI-3 rev 1.           Some later commands are also implemented. */	outbuf[2] = 3;	outbuf[3] = 2; /* Format 2 */	outbuf[4] = 31;        /* Sync data transfer and TCQ.  */        outbuf[7] = 0x10 | (s->tcq ? 0x02 : 0);	r->buf_len = 36;	break;    case 0x16:        DPRINTF("Reserve(6)\n");        if (buf[1] & 1)            goto fail;        break;    case 0x17:        DPRINTF("Release(6)\n");        if (buf[1] & 1)            goto fail;        break;    case 0x1a:    case 0x5a:        {            uint8_t *p;            int page;            page = buf[2] & 0x3f;            DPRINTF("Mode Sense (page %d, len %d)\n", page, len);            p = outbuf;            memset(p, 0, 4);            outbuf[1] = 0; /* Default media type.  */            outbuf[3] = 0; /* Block descriptor length.  */            if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) {                outbuf[2] = 0x80; /* Readonly.  */            }            p += 4;            if ((page == 8 || page == 0x3f)) {                /* Caching page.  */                memset(p,0,20);                p[0] = 8;                p[1] = 0x12;                p[2] = 4; /* WCE */                p += 20;            }            if ((page == 0x3f || page == 0x2a)                    && (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM)) {                /* CD Capabilities and Mechanical Status page. */                p[0] = 0x2a;                p[1] = 0x14;                p[2] = 3; // CD-R & CD-RW read                p[3] = 0; // Writing not supported                p[4] = 0x7f; /* Audio, composite, digital out,                                         mode 2 form 1&2, multi session */                p[5] = 0xff; /* CD DA, DA accurate, RW supported,                                         RW corrected, C2 errors, ISRC,                                         UPC, Bar code */                p[6] = 0x2d | (bdrv_is_locked(s->bdrv)? 2 : 0);                /* Locking supported, jumper present, eject, tray */                p[7] = 0; /* no volume & mute control, no                                      changer */                p[8] = (50 * 176) >> 8; // 50x read speed                p[9] = (50 * 176) & 0xff;                p[10] = 0 >> 8; // No volume                p[11] = 0 & 0xff;                p[12] = 2048 >> 8; // 2M buffer                p[13] = 2048 & 0xff;                p[14] = (16 * 176) >> 8; // 16x read speed current                p[15] = (16 * 176) & 0xff;                p[18] = (16 * 176) >> 8; // 16x write speed                p[19] = (16 * 176) & 0xff;                p[20] = (16 * 176) >> 8; // 16x write speed current                p[21] = (16 * 176) & 0xff;                p += 22;            }            r->buf_len = p - outbuf;            outbuf[0] = r->buf_len - 4;            if (r->buf_len > len)                r->buf_len = len;        }        break;    case 0x1b:        DPRINTF("Start Stop Unit\n");	break;    case 0x1e:        DPRINTF("Prevent Allow Medium Removal (prevent = %d)\n", buf[4] & 3);        bdrv_set_locked(s->bdrv, buf[4] & 1);	break;    case 0x25:	DPRINTF("Read Capacity\n");        /* The normal LEN field for this command is zero.  */	memset(outbuf, 0, 8);	bdrv_get_geometry(s->bdrv, &nb_sectors);        /* Returned value is the address of the last sector.  */        if (nb_sectors) {            nb_sectors--;            outbuf[0] = (nb_sectors >> 24) & 0xff;            outbuf[1] = (nb_sectors >> 16) & 0xff;            outbuf[2] = (nb_sectors >> 8) & 0xff;            outbuf[3] = nb_sectors & 0xff;            outbuf[4] = 0;            outbuf[5] = 0;            outbuf[6] = s->cluster_size * 2;            outbuf[7] = 0;            r->buf_len = 8;        } else {            scsi_command_complete(r, SENSE_NOT_READY);            return 0;        }	break;    case 0x08:    case 0x28:        DPRINTF("Read (sector %d, count %d)\n", lba, len);        r->sector = lba * s->cluster_size;        r->sector_count = len * s->cluster_size;        break;    case 0x0a:    case 0x2a:        DPRINTF("Write (sector %d, count %d)\n", lba, len);        r->sector = lba * s->cluster_size;        r->sector_count = len * s->cluster_size;        is_write = 1;        break;    case 0x35:        DPRINTF("Synchronise cache (sector %d, count %d)\n", lba, len);	bdrv_aio_flush(s->bdrv, scsi_flush_cb, r);	return 0;    case 0x43:        {            int start_track, format, msf, toclen;            msf = buf[1] & 2;            format = buf[2] & 0xf;            start_track = buf[6];            bdrv_get_geometry(s->bdrv, &nb_sectors);            DPRINTF("Read TOC (track %d format %d msf %d)\n", start_track, format, msf >> 1);            switch(format) {            case 0:                toclen = cdrom_read_toc(nb_sectors, outbuf, msf, start_track);                break;            case 1:                /* multi session : only a single session defined */                toclen = 12;                memset(outbuf, 0, 12);                outbuf[1] = 0x0a;                outbuf[2] = 0x01;                outbuf[3] = 0x01;                break;            case 2:                toclen = cdrom_read_toc_raw(nb_sectors, outbuf, msf, start_track);                break;            default:                goto error_cmd;            }            if (toclen > 0) {                if (len > toclen)                  len = toclen;                r->buf_len = len;                break;            }        error_cmd:            DPRINTF("Read TOC error\n");            goto fail;        }    case 0x46:        DPRINTF("Get Configuration (rt %d, maxlen %d)\n", buf[1] & 3, len);        memset(outbuf, 0, 8);        /* ??? This shoud probably return much more information.  For now           just return the basic header indicating the CD-ROM profile.  */        outbuf[7] = 8; // CD-ROM        r->buf_len = 8;        break;    case 0x56:        DPRINTF("Reserve(10)\n");        if (buf[1] & 3)            goto fail;        break;    case 0x57:        DPRINTF("Release(10)\n");        if (buf[1] & 3)            goto fail;        break;    case 0xa0:        DPRINTF("Report LUNs (len %d)\n", len);        if (len < 16)            goto fail;        memset(outbuf, 0, 16);        outbuf[3] = 8;        r->buf_len = 16;        break;    default:	DPRINTF("Unknown SCSI command (%2.2x)\n", buf[0]);    fail:        scsi_command_complete(r, SENSE_ILLEGAL_REQUEST);	return 0;    }    if (r->sector_count == 0 && r->buf_len == 0) {        scsi_command_complete(r, SENSE_NO_SENSE);    }    len = r->sector_count * 512 + r->buf_len;    if (is_write) {        return -len;    } else {        if (!r->sector_count)            r->sector_count = -1;        return len;    }}static void scsi_destroy(SCSIDevice *d){    qemu_free(d->state);    qemu_free(d);}SCSIDevice *scsi_disk_init(BlockDriverState *bdrv, int tcq,                           scsi_completionfn completion, void *opaque){    SCSIDevice *d;    SCSIDeviceState *s;    s = (SCSIDeviceState *)qemu_mallocz(sizeof(SCSIDeviceState));    s->bdrv = bdrv;    s->tcq = tcq;    s->completion = completion;    s->opaque = opaque;    if (bdrv_get_type_hint(s->bdrv) == BDRV_TYPE_CDROM) {        s->cluster_size = 4;    } else {        s->cluster_size = 1;    }    d = (SCSIDevice *)qemu_mallocz(sizeof(SCSIDevice));    d->state = s;    d->destroy = scsi_destroy;    d->send_command = scsi_send_command;    d->read_data = scsi_read_data;    d->write_data = scsi_write_data;    d->cancel_io = scsi_cancel_io;    d->get_buf = scsi_get_buf;    return d;}

⌨️ 快捷键说明

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