11ae08745Sheppo /* 21ae08745Sheppo * CDDL HEADER START 31ae08745Sheppo * 41ae08745Sheppo * The contents of this file are subject to the terms of the 51ae08745Sheppo * Common Development and Distribution License (the "License"). 61ae08745Sheppo * You may not use this file except in compliance with the License. 71ae08745Sheppo * 81ae08745Sheppo * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 91ae08745Sheppo * or http://www.opensolaris.org/os/licensing. 101ae08745Sheppo * See the License for the specific language governing permissions 111ae08745Sheppo * and limitations under the License. 121ae08745Sheppo * 131ae08745Sheppo * When distributing Covered Code, include this CDDL HEADER in each 141ae08745Sheppo * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 151ae08745Sheppo * If applicable, add the following below this CDDL HEADER, with the 161ae08745Sheppo * fields enclosed by brackets "[]" replaced with your own identifying 171ae08745Sheppo * information: Portions Copyright [yyyy] [name of copyright owner] 181ae08745Sheppo * 191ae08745Sheppo * CDDL HEADER END 201ae08745Sheppo */ 211ae08745Sheppo 221ae08745Sheppo /* 231ae08745Sheppo * Copyright 2006 Sun Microsystems, Inc. All rights reserved. 241ae08745Sheppo * Use is subject to license terms. 251ae08745Sheppo */ 261ae08745Sheppo 271ae08745Sheppo #ifndef _VDC_H 281ae08745Sheppo #define _VDC_H 291ae08745Sheppo 301ae08745Sheppo #pragma ident "%Z%%M% %I% %E% SMI" 311ae08745Sheppo 321ae08745Sheppo /* 331ae08745Sheppo * Virtual disk client implementation definitions 341ae08745Sheppo */ 351ae08745Sheppo 361ae08745Sheppo #include <sys/sysmacros.h> 371ae08745Sheppo #include <sys/note.h> 381ae08745Sheppo 391ae08745Sheppo #include <sys/ldc.h> 401ae08745Sheppo #include <sys/vio_mailbox.h> 411ae08745Sheppo #include <sys/vdsk_mailbox.h> 421ae08745Sheppo #include <sys/vdsk_common.h> 431ae08745Sheppo 441ae08745Sheppo #ifdef __cplusplus 451ae08745Sheppo extern "C" { 461ae08745Sheppo #endif 471ae08745Sheppo 481ae08745Sheppo #define VDC_DRIVER_NAME "vdc" 491ae08745Sheppo 501ae08745Sheppo /* 511ae08745Sheppo * Bit-field values to indicate if parts of the vdc driver are initialised. 521ae08745Sheppo */ 531ae08745Sheppo #define VDC_SOFT_STATE 0x0001 541ae08745Sheppo #define VDC_LOCKS 0x0002 551ae08745Sheppo #define VDC_MINOR 0x0004 561ae08745Sheppo #define VDC_THREAD 0x0008 571ae08745Sheppo #define VDC_LDC 0x0010 581ae08745Sheppo #define VDC_LDC_INIT 0x0020 591ae08745Sheppo #define VDC_LDC_CB 0x0040 601ae08745Sheppo #define VDC_LDC_OPEN 0x0080 611ae08745Sheppo #define VDC_DRING_INIT 0x0100 /* The DRing was created */ 621ae08745Sheppo #define VDC_DRING_BOUND 0x0200 /* The DRing was bound to an LDC channel */ 631ae08745Sheppo #define VDC_DRING_LOCAL 0x0400 /* The local private DRing was allocated */ 641ae08745Sheppo #define VDC_DRING_ENTRY 0x0800 /* At least one DRing entry was initialised */ 651ae08745Sheppo #define VDC_DRING (VDC_DRING_INIT | VDC_DRING_BOUND | \ 661ae08745Sheppo VDC_DRING_LOCAL | VDC_DRING_ENTRY) 671ae08745Sheppo #define VDC_HANDSHAKE 0x1000 /* Indicates if a handshake is in progress */ 681ae08745Sheppo #define VDC_HANDSHAKE_STOP 0x2000 /* stop further handshakes */ 691ae08745Sheppo 701ae08745Sheppo /* 711ae08745Sheppo * Definitions of strings to be used to create device node properties. 721ae08745Sheppo * (vdc uses the capitalised versions of these properties as they are 64-bit) 731ae08745Sheppo */ 741ae08745Sheppo #define VDC_NBLOCKS_PROP_NAME "Nblocks" 751ae08745Sheppo #define VDC_SIZE_PROP_NAME "Size" 761ae08745Sheppo 771ae08745Sheppo /* 781ae08745Sheppo * Definitions of MD nodes/properties. 791ae08745Sheppo */ 801ae08745Sheppo #define VDC_MD_CHAN_NAME "channel-endpoint" 811ae08745Sheppo #define VDC_MD_VDEV_NAME "virtual-device" 821ae08745Sheppo #define VDC_MD_DISK_NAME "disk" 831ae08745Sheppo #define VDC_MD_CFG_HDL "cfg-handle" 841ae08745Sheppo #define VDC_ID_PROP "id" 851ae08745Sheppo 861ae08745Sheppo /* 87e1ebb9ecSlm66018 * Definition of actions to be carried out when processing the sequence ID 88e1ebb9ecSlm66018 * of a message received from the vDisk server. The function verifying the 89e1ebb9ecSlm66018 * sequence number checks the 'seq_num_xxx' fields in the soft state and 90e1ebb9ecSlm66018 * returns whether the message should be processed (VDC_SEQ_NUM_TODO) or 91e1ebb9ecSlm66018 * whether it was it was previously processed (VDC_SEQ_NUM_SKIP). 92e1ebb9ecSlm66018 */ 93e1ebb9ecSlm66018 #define VDC_SEQ_NUM_INVALID -1 /* Error */ 94e1ebb9ecSlm66018 #define VDC_SEQ_NUM_SKIP 0 /* Request already processed */ 95e1ebb9ecSlm66018 #define VDC_SEQ_NUM_TODO 1 /* Request needs processing */ 96e1ebb9ecSlm66018 97e1ebb9ecSlm66018 /* 981ae08745Sheppo * Scheme to store the instance number and the slice number in the minor number. 991ae08745Sheppo * (Uses the same format and definitions as the sd(7D) driver) 1001ae08745Sheppo */ 1011ae08745Sheppo #define VD_MAKE_DEV(instance, minor) ((instance << SDUNIT_SHIFT) | minor) 1021ae08745Sheppo 1031ae08745Sheppo /* 1041ae08745Sheppo * variables controlling how long to wait before timing out and how many 1051ae08745Sheppo * retries to attempt before giving up when communicating with vds. 106e1ebb9ecSlm66018 * 107e1ebb9ecSlm66018 * These values need to be sufficiently large so that a guest can survive 108e1ebb9ecSlm66018 * the reboot of the service domain. 1091ae08745Sheppo */ 110e1ebb9ecSlm66018 #define VDC_RETRIES 10 1111ae08745Sheppo 1121ae08745Sheppo #define VDC_USEC_TIMEOUT_MIN (30 * MICROSEC) /* 30 sec */ 1131ae08745Sheppo 114e1ebb9ecSlm66018 /* 115e1ebb9ecSlm66018 * This macro returns the number of Hz that the vdc driver should wait before 116e1ebb9ecSlm66018 * a timeout is triggered. The 'timeout' parameter specifiecs the wait 117e1ebb9ecSlm66018 * time in Hz. The 'mul' parameter allows for a multiplier to be 118e1ebb9ecSlm66018 * specified allowing for a backoff to be implemented (e.g. using the 119e1ebb9ecSlm66018 * retry number as a multiplier) where the wait time will get longer if 120e1ebb9ecSlm66018 * there is no response on the previous retry. 121e1ebb9ecSlm66018 */ 122e1ebb9ecSlm66018 #define VD_GET_TIMEOUT_HZ(timeout, mul) \ 123e1ebb9ecSlm66018 (ddi_get_lbolt() + ((timeout) * MAX(1, (mul)))) 1241ae08745Sheppo 1251ae08745Sheppo /* 1261ae08745Sheppo * Macros to manipulate Descriptor Ring variables in the soft state 1271ae08745Sheppo * structure. 1281ae08745Sheppo */ 129e1ebb9ecSlm66018 #define VDC_GET_NEXT_REQ_ID(vdc) ((vdc)->req_id++) 1301ae08745Sheppo 1311ae08745Sheppo #define VDC_GET_DRING_ENTRY_PTR(vdc, idx) \ 132e1ebb9ecSlm66018 (vd_dring_entry_t *)((vdc)->dring_mem_info.vaddr + \ 133e1ebb9ecSlm66018 (idx * (vdc)->dring_entry_size)) 1341ae08745Sheppo 1351ae08745Sheppo #define VDC_MARK_DRING_ENTRY_FREE(vdc, idx) \ 1361ae08745Sheppo { \ 1371ae08745Sheppo vd_dring_entry_t *dep = NULL; \ 1381ae08745Sheppo ASSERT(vdc != NULL); \ 139e1ebb9ecSlm66018 ASSERT((idx >= 0) && (idx < vdc->dring_len)); \ 1401ae08745Sheppo ASSERT(vdc->dring_mem_info.vaddr != NULL); \ 1411ae08745Sheppo dep = (vd_dring_entry_t *)(vdc->dring_mem_info.vaddr + \ 1421ae08745Sheppo (idx * vdc->dring_entry_size)); \ 1431ae08745Sheppo ASSERT(dep != NULL); \ 1441ae08745Sheppo dep->hdr.dstate = VIO_DESC_FREE; \ 1451ae08745Sheppo } 1461ae08745Sheppo 1471ae08745Sheppo /* Initialise the Session ID and Sequence Num in the DRing msg */ 1481ae08745Sheppo #define VDC_INIT_DRING_DATA_MSG_IDS(dmsg, vdc) \ 1491ae08745Sheppo ASSERT(vdc != NULL); \ 1501ae08745Sheppo dmsg.tag.vio_sid = vdc->session_id; \ 1510a55fbb7Slm66018 dmsg.seq_num = vdc->seq_num; 1521ae08745Sheppo 1531ae08745Sheppo /* 1541ae08745Sheppo * The states the message processing thread can be in. 1551ae08745Sheppo */ 1561ae08745Sheppo typedef enum vdc_thr_state { 1571ae08745Sheppo VDC_THR_RUNNING, /* thread is running & ready to process */ 1581ae08745Sheppo VDC_THR_STOP, /* The detach func signals the thread to stop */ 1591ae08745Sheppo VDC_THR_DONE /* Thread has exited */ 1601ae08745Sheppo } vdc_thr_state_t; 1611ae08745Sheppo 1621ae08745Sheppo /* 1631ae08745Sheppo * Local Descriptor Ring entry 1641ae08745Sheppo * 1651ae08745Sheppo * vdc creates a Local (private) descriptor ring the same size as the 1661ae08745Sheppo * public descriptor ring it exports to vds. 1671ae08745Sheppo */ 1681ae08745Sheppo typedef struct vdc_local_desc { 1691ae08745Sheppo kmutex_t lock; /* protects all fields */ 1701ae08745Sheppo kcondvar_t cv; /* indicate processing done */ 1711ae08745Sheppo int flags; /* Dring entry state, etc */ 1721ae08745Sheppo int operation; /* VD_OP_xxx to be performed */ 1731ae08745Sheppo caddr_t addr; /* addr passed in by consumer */ 1741ae08745Sheppo caddr_t align_addr; /* used if addr non-aligned */ 1751ae08745Sheppo struct buf *buf; /* buf passed to strategy() */ 1761ae08745Sheppo ldc_mem_handle_t desc_mhdl; /* Mem handle of buf */ 1771ae08745Sheppo vd_dring_entry_t *dep; /* public Dring Entry Pointer */ 1781ae08745Sheppo } vdc_local_desc_t; 1791ae08745Sheppo 1801ae08745Sheppo /* 1811ae08745Sheppo * vdc soft state structure 1821ae08745Sheppo */ 1831ae08745Sheppo typedef struct vdc { 1841ae08745Sheppo kmutex_t attach_lock; /* used by CV which waits in attach */ 1851ae08745Sheppo kcondvar_t attach_cv; /* signal when attach can finish */ 1861ae08745Sheppo 1871ae08745Sheppo kmutex_t lock; /* protects next 2 sections of vars */ 1881ae08745Sheppo kcondvar_t cv; /* signal when upper layers can send */ 1891ae08745Sheppo 1901ae08745Sheppo int initialized; /* keeps track of what's init'ed */ 1910a55fbb7Slm66018 int hshake_cnt; /* number of failed handshakes */ 1921ae08745Sheppo int open; /* count of outstanding opens */ 1931ae08745Sheppo int dkio_flush_pending; /* # outstanding DKIO flushes */ 1941ae08745Sheppo 1951ae08745Sheppo uint64_t session_id; /* common ID sent with all messages */ 1961ae08745Sheppo uint64_t seq_num; /* most recent sequence num generated */ 1971ae08745Sheppo uint64_t seq_num_reply; /* Last seq num ACK/NACK'ed by vds */ 1981ae08745Sheppo uint64_t req_id; /* Most recent Request ID generated */ 199e1ebb9ecSlm66018 uint64_t req_id_proc; /* Last request ID processed by vdc */ 2001ae08745Sheppo vd_state_t state; /* Current handshake state */ 201e1ebb9ecSlm66018 202e1ebb9ecSlm66018 dev_info_t *dip; /* device info pointer */ 203e1ebb9ecSlm66018 int instance; /* driver instance number */ 204e1ebb9ecSlm66018 vio_ver_t ver; /* version number agreed with server */ 2051ae08745Sheppo vd_disk_type_t vdisk_type; /* type of device/disk being imported */ 206*4bac2208Snarayan vd_disk_label_t vdisk_label; /* label type of device/disk imported */ 207*4bac2208Snarayan uint64_t vdisk_size; /* device size in blocks */ 2081ae08745Sheppo uint64_t max_xfer_sz; /* maximum block size of a descriptor */ 2091ae08745Sheppo uint64_t block_size; /* device block size used */ 2100a55fbb7Slm66018 struct dk_label *label; /* structure to store disk label */ 2111ae08745Sheppo struct dk_cinfo *cinfo; /* structure to store DKIOCINFO data */ 2121ae08745Sheppo struct dk_minfo *minfo; /* structure for DKIOCGMEDIAINFO data */ 2131ae08745Sheppo struct vtoc *vtoc; /* structure to store VTOC data */ 214*4bac2208Snarayan ddi_devid_t devid; /* device id */ 2151ae08745Sheppo 2161ae08745Sheppo /* 2171ae08745Sheppo * The mutex 'msg_proc_lock' protects the following group of fields. 2181ae08745Sheppo * 2191ae08745Sheppo * The callback function checks to see if LDC triggered it due to 2201ae08745Sheppo * there being data available and the callback will signal to 2211ae08745Sheppo * the message processing thread waiting on 'msg_proc_cv'. 2221ae08745Sheppo */ 2231ae08745Sheppo kmutex_t msg_proc_lock; 2241ae08745Sheppo kcondvar_t msg_proc_cv; 2251ae08745Sheppo boolean_t msg_pending; 2261ae08745Sheppo vdc_thr_state_t msg_proc_thr_state; 2271ae08745Sheppo kthread_t *msg_proc_thr_id; 2281ae08745Sheppo 2291ae08745Sheppo /* 2301ae08745Sheppo * The mutex 'dring_lock' protects the following group of fields. 2311ae08745Sheppo */ 2321ae08745Sheppo kmutex_t dring_lock; 2331ae08745Sheppo ldc_mem_info_t dring_mem_info; 2341ae08745Sheppo uint_t dring_curr_idx; 235e1ebb9ecSlm66018 uint_t dring_proc_idx; 2361ae08745Sheppo uint32_t dring_len; 237e1ebb9ecSlm66018 uint32_t dring_max_cookies; 2381ae08745Sheppo uint32_t dring_cookie_count; 2391ae08745Sheppo uint32_t dring_entry_size; 240e1ebb9ecSlm66018 boolean_t dring_notify_server; 2411ae08745Sheppo ldc_mem_cookie_t *dring_cookie; 2421ae08745Sheppo uint64_t dring_ident; 2431ae08745Sheppo 2441ae08745Sheppo vdc_local_desc_t *local_dring; 2451ae08745Sheppo 2461ae08745Sheppo uint64_t ldc_id; 2471ae08745Sheppo ldc_status_t ldc_state; 2481ae08745Sheppo ldc_handle_t ldc_handle; 2491ae08745Sheppo ldc_dring_handle_t ldc_dring_hdl; 2501ae08745Sheppo } vdc_t; 2511ae08745Sheppo 2521ae08745Sheppo /* 2531ae08745Sheppo * Debugging macros 2541ae08745Sheppo */ 2551ae08745Sheppo #ifdef DEBUG 2561ae08745Sheppo extern int vdc_msglevel; 2571ae08745Sheppo 258e1ebb9ecSlm66018 #define DMSG(err_level, format, ...) \ 259e1ebb9ecSlm66018 do { \ 260e1ebb9ecSlm66018 if (vdc_msglevel > err_level) \ 261e1ebb9ecSlm66018 cmn_err(CE_CONT, "?%s"format, __func__, __VA_ARGS__);\ 262e1ebb9ecSlm66018 _NOTE(CONSTANTCONDITION) \ 263e1ebb9ecSlm66018 } while (0); 2641ae08745Sheppo 2651ae08745Sheppo #define VDC_DUMP_DRING_MSG(dmsgp) \ 266e1ebb9ecSlm66018 DMSG(0, "sq:%lu start:%d end:%d ident:%lu\n", \ 2671ae08745Sheppo dmsgp->seq_num, dmsgp->start_idx, \ 2681ae08745Sheppo dmsgp->end_idx, dmsgp->dring_ident); 2691ae08745Sheppo 2701ae08745Sheppo #else /* !DEBUG */ 271e1ebb9ecSlm66018 #define DMSG(err_level, ...) 2721ae08745Sheppo #define VDC_DUMP_DRING_MSG(dmsgp) 2731ae08745Sheppo 2741ae08745Sheppo #endif /* !DEBUG */ 2751ae08745Sheppo 2761ae08745Sheppo #ifdef __cplusplus 2771ae08745Sheppo } 2781ae08745Sheppo #endif 2791ae08745Sheppo 2801ae08745Sheppo #endif /* _VDC_H */ 281