1a9643ea8Slogwang /*- 2*22ce4affSfengbojiang * SPDX-License-Identifier: BSD-2-Clause-FreeBSD 3*22ce4affSfengbojiang * 4a9643ea8Slogwang * Copyright (c) 2015 Spectra Logic Corporation 5a9643ea8Slogwang * All rights reserved. 6a9643ea8Slogwang * 7a9643ea8Slogwang * Redistribution and use in source and binary forms, with or without 8a9643ea8Slogwang * modification, are permitted provided that the following conditions 9a9643ea8Slogwang * are met: 10a9643ea8Slogwang * 1. Redistributions of source code must retain the above copyright 11a9643ea8Slogwang * notice, this list of conditions, and the following disclaimer, 12a9643ea8Slogwang * without modification. 13a9643ea8Slogwang * 2. Redistributions in binary form must reproduce at minimum a disclaimer 14a9643ea8Slogwang * substantially similar to the "NO WARRANTY" disclaimer below 15a9643ea8Slogwang * ("Disclaimer") and any redistribution must be conditioned upon 16a9643ea8Slogwang * including a substantially similar Disclaimer requirement for further 17a9643ea8Slogwang * binary redistribution. 18a9643ea8Slogwang * 19a9643ea8Slogwang * NO WARRANTY 20a9643ea8Slogwang * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 21a9643ea8Slogwang * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22a9643ea8Slogwang * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR 23a9643ea8Slogwang * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 24a9643ea8Slogwang * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 25a9643ea8Slogwang * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 26a9643ea8Slogwang * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 27a9643ea8Slogwang * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, 28a9643ea8Slogwang * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 29a9643ea8Slogwang * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 30a9643ea8Slogwang * POSSIBILITY OF SUCH DAMAGES. 31a9643ea8Slogwang * 32a9643ea8Slogwang * Authors: Ken Merry (Spectra Logic Corporation) 33a9643ea8Slogwang * 34a9643ea8Slogwang * $FreeBSD$ 35a9643ea8Slogwang */ 36a9643ea8Slogwang 37a9643ea8Slogwang #ifndef _SYS_DISK_ZONE_H_ 38a9643ea8Slogwang #define _SYS_DISK_ZONE_H_ 39a9643ea8Slogwang 40a9643ea8Slogwang /* 41a9643ea8Slogwang * Interface for Zone-based disks. This allows managing devices that 42a9643ea8Slogwang * conform to the SCSI Zoned Block Commands (ZBC) and ATA Zoned ATA Command 43a9643ea8Slogwang * Set (ZAC) specifications. Devices using these command sets are 44a9643ea8Slogwang * currently (October 2015) hard drives using Shingled Magnetic Recording 45a9643ea8Slogwang * (SMR). 46a9643ea8Slogwang */ 47a9643ea8Slogwang 48a9643ea8Slogwang /* 49a9643ea8Slogwang * There are currently three types of zoned devices: 50a9643ea8Slogwang * 51a9643ea8Slogwang * Drive Managed: 52a9643ea8Slogwang * Drive Managed drives look and act just like a standard random access 53a9643ea8Slogwang * block device, but underneath, the drive reads and writes the bulk of 54a9643ea8Slogwang * its capacity using SMR zones. Sequential writes will yield better 55a9643ea8Slogwang * performance, but writing sequentially is not required. 56a9643ea8Slogwang * 57a9643ea8Slogwang * Host Aware: 58a9643ea8Slogwang * Host Aware drives expose the underlying zone layout via SCSI or ATA 59a9643ea8Slogwang * commands and allow the host to manage the zone conditions. The host 60a9643ea8Slogwang * is not required to manage the zones on the drive, though. Sequential 61a9643ea8Slogwang * writes will yield better performance in Sequential Write Preferred 62a9643ea8Slogwang * zones, but the host can write randomly in those zones. 63a9643ea8Slogwang * 64a9643ea8Slogwang * Host Managed: 65a9643ea8Slogwang * Host Managed drives expose the underlying zone layout via SCSI or ATA 66a9643ea8Slogwang * commands. The host is required to access the zones according to the 67a9643ea8Slogwang * rules described by the zone layout. Any commands that violate the 68a9643ea8Slogwang * rules will be returned with an error. 69a9643ea8Slogwang */ 70a9643ea8Slogwang struct disk_zone_disk_params { 71a9643ea8Slogwang uint32_t zone_mode; 72a9643ea8Slogwang #define DISK_ZONE_MODE_NONE 0x00 73a9643ea8Slogwang #define DISK_ZONE_MODE_HOST_AWARE 0x01 74a9643ea8Slogwang #define DISK_ZONE_MODE_DRIVE_MANAGED 0x02 75a9643ea8Slogwang #define DISK_ZONE_MODE_HOST_MANAGED 0x04 76a9643ea8Slogwang uint64_t flags; 77a9643ea8Slogwang #define DISK_ZONE_DISK_URSWRZ 0x001 78a9643ea8Slogwang #define DISK_ZONE_OPT_SEQ_SET 0x002 79a9643ea8Slogwang #define DISK_ZONE_OPT_NONSEQ_SET 0x004 80a9643ea8Slogwang #define DISK_ZONE_MAX_SEQ_SET 0x008 81a9643ea8Slogwang #define DISK_ZONE_RZ_SUP 0x010 82a9643ea8Slogwang #define DISK_ZONE_OPEN_SUP 0x020 83a9643ea8Slogwang #define DISK_ZONE_CLOSE_SUP 0x040 84a9643ea8Slogwang #define DISK_ZONE_FINISH_SUP 0x080 85a9643ea8Slogwang #define DISK_ZONE_RWP_SUP 0x100 86a9643ea8Slogwang #define DISK_ZONE_CMD_SUP_MASK 0x1f0 87a9643ea8Slogwang uint64_t optimal_seq_zones; 88a9643ea8Slogwang uint64_t optimal_nonseq_zones; 89a9643ea8Slogwang uint64_t max_seq_zones; 90a9643ea8Slogwang }; 91a9643ea8Slogwang 92a9643ea8Slogwang /* 93a9643ea8Slogwang * Used for reset write pointer, open, close and finish. 94a9643ea8Slogwang */ 95a9643ea8Slogwang struct disk_zone_rwp { 96a9643ea8Slogwang uint64_t id; 97a9643ea8Slogwang uint8_t flags; 98a9643ea8Slogwang #define DISK_ZONE_RWP_FLAG_NONE 0x00 99a9643ea8Slogwang #define DISK_ZONE_RWP_FLAG_ALL 0x01 100a9643ea8Slogwang }; 101a9643ea8Slogwang 102a9643ea8Slogwang /* 103a9643ea8Slogwang * Report Zones header. All of these values are passed out. 104a9643ea8Slogwang */ 105a9643ea8Slogwang struct disk_zone_rep_header { 106a9643ea8Slogwang uint8_t same; 107a9643ea8Slogwang #define DISK_ZONE_SAME_ALL_DIFFERENT 0x0 /* Lengths and types vary */ 108a9643ea8Slogwang #define DISK_ZONE_SAME_ALL_SAME 0x1 /* Lengths and types the same */ 109a9643ea8Slogwang #define DISK_ZONE_SAME_LAST_DIFFERENT 0x2 /* Types same, last len varies */ 110a9643ea8Slogwang #define DISK_ZONE_SAME_TYPES_DIFFERENT 0x3 /* Types vary, length the same */ 111a9643ea8Slogwang uint64_t maximum_lba; 112a9643ea8Slogwang /* 113a9643ea8Slogwang * XXX KDM padding space may not be a good idea inside the bio. 114a9643ea8Slogwang */ 115a9643ea8Slogwang uint8_t reserved[64]; 116a9643ea8Slogwang }; 117a9643ea8Slogwang 118a9643ea8Slogwang /* 119a9643ea8Slogwang * Report Zones entry. Note that the zone types, conditions, and flags 120a9643ea8Slogwang * are mapped directly from the SCSI/ATA flag values. Any additional 121a9643ea8Slogwang * SCSI/ATA zone types or conditions or flags that are defined in the 122a9643ea8Slogwang * future could result in additional values that are not yet defined here. 123a9643ea8Slogwang */ 124a9643ea8Slogwang struct disk_zone_rep_entry { 125a9643ea8Slogwang uint8_t zone_type; 126a9643ea8Slogwang #define DISK_ZONE_TYPE_CONVENTIONAL 0x01 127a9643ea8Slogwang #define DISK_ZONE_TYPE_SEQ_REQUIRED 0x02 /* Host Managed */ 128a9643ea8Slogwang #define DISK_ZONE_TYPE_SEQ_PREFERRED 0x03 /* Host Aware */ 129a9643ea8Slogwang uint8_t zone_condition; 130a9643ea8Slogwang #define DISK_ZONE_COND_NOT_WP 0x00 131a9643ea8Slogwang #define DISK_ZONE_COND_EMPTY 0x01 132a9643ea8Slogwang #define DISK_ZONE_COND_IMPLICIT_OPEN 0x02 133a9643ea8Slogwang #define DISK_ZONE_COND_EXPLICIT_OPEN 0x03 134a9643ea8Slogwang #define DISK_ZONE_COND_CLOSED 0x04 135a9643ea8Slogwang #define DISK_ZONE_COND_READONLY 0x0D 136a9643ea8Slogwang #define DISK_ZONE_COND_FULL 0x0E 137a9643ea8Slogwang #define DISK_ZONE_COND_OFFLINE 0x0F 138a9643ea8Slogwang uint8_t zone_flags; 139a9643ea8Slogwang #define DISK_ZONE_FLAG_RESET 0x01 /* Zone needs RWP */ 140a9643ea8Slogwang #define DISK_ZONE_FLAG_NON_SEQ 0x02 /* Zone accssessed nonseq */ 141a9643ea8Slogwang uint64_t zone_length; 142a9643ea8Slogwang uint64_t zone_start_lba; 143a9643ea8Slogwang uint64_t write_pointer_lba; 144a9643ea8Slogwang /* XXX KDM padding space may not be a good idea inside the bio */ 145a9643ea8Slogwang uint8_t reserved[32]; 146a9643ea8Slogwang }; 147a9643ea8Slogwang 148a9643ea8Slogwang struct disk_zone_report { 149a9643ea8Slogwang uint64_t starting_id; /* Passed In */ 150a9643ea8Slogwang uint8_t rep_options; /* Passed In */ 151a9643ea8Slogwang #define DISK_ZONE_REP_ALL 0x00 152a9643ea8Slogwang #define DISK_ZONE_REP_EMPTY 0x01 153a9643ea8Slogwang #define DISK_ZONE_REP_IMP_OPEN 0x02 154a9643ea8Slogwang #define DISK_ZONE_REP_EXP_OPEN 0x03 155a9643ea8Slogwang #define DISK_ZONE_REP_CLOSED 0x04 156a9643ea8Slogwang #define DISK_ZONE_REP_FULL 0x05 157a9643ea8Slogwang #define DISK_ZONE_REP_READONLY 0x06 158a9643ea8Slogwang #define DISK_ZONE_REP_OFFLINE 0x07 159a9643ea8Slogwang #define DISK_ZONE_REP_RWP 0x10 160a9643ea8Slogwang #define DISK_ZONE_REP_NON_SEQ 0x11 161a9643ea8Slogwang #define DISK_ZONE_REP_NON_WP 0x3F 162a9643ea8Slogwang struct disk_zone_rep_header header; 163a9643ea8Slogwang uint32_t entries_allocated; /* Passed In */ 164a9643ea8Slogwang uint32_t entries_filled; /* Passed Out */ 165a9643ea8Slogwang uint32_t entries_available; /* Passed Out */ 166a9643ea8Slogwang struct disk_zone_rep_entry *entries; 167a9643ea8Slogwang }; 168a9643ea8Slogwang 169a9643ea8Slogwang union disk_zone_params { 170a9643ea8Slogwang struct disk_zone_disk_params disk_params; 171a9643ea8Slogwang struct disk_zone_rwp rwp; 172a9643ea8Slogwang struct disk_zone_report report; 173a9643ea8Slogwang }; 174a9643ea8Slogwang 175a9643ea8Slogwang struct disk_zone_args { 176a9643ea8Slogwang uint8_t zone_cmd; 177a9643ea8Slogwang #define DISK_ZONE_OPEN 0x00 178a9643ea8Slogwang #define DISK_ZONE_CLOSE 0x01 179a9643ea8Slogwang #define DISK_ZONE_FINISH 0x02 180a9643ea8Slogwang #define DISK_ZONE_REPORT_ZONES 0x03 181a9643ea8Slogwang #define DISK_ZONE_RWP 0x04 182a9643ea8Slogwang #define DISK_ZONE_GET_PARAMS 0x05 183a9643ea8Slogwang union disk_zone_params zone_params; 184a9643ea8Slogwang }; 185a9643ea8Slogwang 186a9643ea8Slogwang #endif /* _SYS_DISK_ZONE_H_ */ 187