VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvHostBase-solaris.cpp@ 64276

Last change on this file since 64276 was 64252, checked in by vboxsync, 8 years ago

Devices/Storage/DrvHost*: Move destroying the host specific resources to the host specific code

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 6.9 KB
Line 
1/* $Id: DrvHostBase-solaris.cpp 64252 2016-10-13 14:20:01Z vboxsync $ */
2/** @file
3 * DrvHostBase - Host base drive access driver, Solaris specifics.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17#define LOG_GROUP LOG_GROUP_DRV_HOST_BASE
18#include <fcntl.h>
19#include <errno.h>
20#include <stropts.h>
21#include <malloc.h>
22#include <sys/dkio.h>
23#include <pwd.h>
24#include <unistd.h>
25#include <syslog.h>
26#ifdef VBOX_WITH_SUID_WRAPPER
27# include <auth_attr.h>
28#endif
29#include <sys/sockio.h>
30#include <sys/scsi/scsi.h>
31
32extern "C" char *getfullblkname(char *);
33
34#include <iprt/file.h>
35#include "DrvHostBase.h"
36
37#ifdef VBOX_WITH_SUID_WRAPPER
38/* These functions would have to go into a separate solaris binary with
39 * the setuid permission set, which would run the user-SCSI ioctl and
40 * return the value. BUT... this might be prohibitively slow.
41 */
42
43/**
44 * Setuid wrapper to gain root access.
45 *
46 * @returns VBox error code.
47 * @param pEffUserID Pointer to effective user ID.
48 */
49static int solarisEnterRootMode(uid_t *pEffUserID)
50{
51 /* Increase privilege if required */
52 if (*pEffUserID != 0)
53 {
54 if (seteuid(0) == 0)
55 {
56 *pEffUserID = 0;
57 return VINF_SUCCESS;
58 }
59 return VERR_PERMISSION_DENIED;
60 }
61 return VINF_SUCCESS;
62}
63
64
65/**
66 * Setuid wrapper to relinquish root access.
67 *
68 * @returns VBox error code.
69 * @param pEffUserID Pointer to effective user ID.
70 */
71static int solarisExitRootMode(uid_t *pEffUserID)
72{
73 /* Get back to user mode. */
74 if (*pEffUserID == 0)
75 {
76 uid_t realID = getuid();
77 if (seteuid(realID) == 0)
78 {
79 *pEffUserID = realID;
80 return VINF_SUCCESS;
81 }
82 return VERR_PERMISSION_DENIED;
83 }
84 return VINF_SUCCESS;
85}
86
87#endif /* VBOX_WITH_SUID_WRAPPER */
88
89DECLHIDDEN(int) drvHostBaseScsiCmdOs(PDRVHOSTBASE pThis, const uint8_t *pbCmd, size_t cbCmd, PDMMEDIATXDIR enmTxDir,
90 void *pvBuf, uint32_t *pcbBuf, uint8_t *pbSense, size_t cbSense, uint32_t cTimeoutMillies)
91{
92 /*
93 * Minimal input validation.
94 */
95 Assert(enmTxDir == PDMMEDIATXDIR_NONE || enmTxDir == PDMMEDIATXDIR_FROM_DEVICE || enmTxDir == PDMMEDIATXDIR_TO_DEVICE);
96 Assert(!pvBuf || pcbBuf);
97 Assert(pvBuf || enmTxDir == PDMMEDIATXDIR_NONE);
98 Assert(pbSense || !cbSense);
99 AssertPtr(pbCmd);
100 Assert(cbCmd <= 16 && cbCmd >= 1);
101
102 int rc = VERR_GENERAL_FAILURE;
103 struct uscsi_cmd usc;
104 union scsi_cdb scdb;
105 memset(&usc, 0, sizeof(struct uscsi_cmd));
106 memset(&scdb, 0, sizeof(scdb));
107
108 switch (enmTxDir)
109 {
110 case PDMMEDIATXDIR_NONE:
111 Assert(*pcbBuf == 0);
112 usc.uscsi_flags = USCSI_READ;
113 /* nothing to do */
114 break;
115
116 case PDMMEDIATXDIR_FROM_DEVICE:
117 Assert(*pcbBuf != 0);
118 /* Make sure that the buffer is clear for commands reading
119 * data. The actually received data may be shorter than what
120 * we expect, and due to the unreliable feedback about how much
121 * data the ioctl actually transferred, it's impossible to
122 * prevent that. Returning previous buffer contents may cause
123 * security problems inside the guest OS, if users can issue
124 * commands to the CDROM device. */
125 memset(pvBuf, '\0', *pcbBuf);
126 usc.uscsi_flags = USCSI_READ;
127 break;
128 case PDMMEDIATXDIR_TO_DEVICE:
129 Assert(*pcbBuf != 0);
130 usc.uscsi_flags = USCSI_WRITE;
131 break;
132 default:
133 AssertMsgFailedReturn(("%d\n", enmTxDir), VERR_INTERNAL_ERROR);
134 }
135 usc.uscsi_flags |= USCSI_RQENABLE;
136 usc.uscsi_rqbuf = (char *)pbSense;
137 usc.uscsi_rqlen = cbSense;
138 usc.uscsi_cdb = (caddr_t)&scdb;
139 usc.uscsi_cdblen = 12;
140 memcpy (usc.uscsi_cdb, pbCmd, usc.uscsi_cdblen);
141 usc.uscsi_bufaddr = (caddr_t)pvBuf;
142 usc.uscsi_buflen = *pcbBuf;
143 usc.uscsi_timeout = (cTimeoutMillies + 999) / 1000;
144
145 /* We need root privileges for user-SCSI under Solaris. */
146#ifdef VBOX_WITH_SUID_WRAPPER
147 uid_t effUserID = geteuid();
148 solarisEnterRootMode(&effUserID); /** @todo check return code when this really works. */
149#endif
150 rc = ioctl(RTFileToNative(pThis->hFileRawDevice), USCSICMD, &usc);
151#ifdef VBOX_WITH_SUID_WRAPPER
152 solarisExitRootMode(&effUserID);
153#endif
154 if (rc < 0)
155 {
156 if (errno == EPERM)
157 return VERR_PERMISSION_DENIED;
158 if (usc.uscsi_status)
159 {
160 rc = RTErrConvertFromErrno(errno);
161 Log2(("%s: error status. rc=%Rrc\n", __FUNCTION__, rc));
162 }
163 }
164 Log2(("%s: after ioctl: residual buflen=%d original buflen=%d\n", __FUNCTION__, usc.uscsi_resid, usc.uscsi_buflen));
165
166 return rc;
167}
168
169DECLHIDDEN(int) drvHostBaseGetMediaSizeOs(PDRVHOSTBASE pThis, uint64_t *pcb)
170{
171 /*
172 * Sun docs suggests using DKIOCGGEOM instead of DKIOCGMEDIAINFO, but
173 * Sun themselves use DKIOCGMEDIAINFO for DVDs/CDs, and use DKIOCGGEOM
174 * for secondary storage devices.
175 */
176 struct dk_minfo MediaInfo;
177 if (ioctl(RTFileToNative(pThis->hFileRawDevice), DKIOCGMEDIAINFO, &MediaInfo) == 0)
178 {
179 *pcb = MediaInfo.dki_capacity * (uint64_t)MediaInfo.dki_lbsize;
180 return VINF_SUCCESS;
181 }
182 return RTFileSeek(pThis->hFileDevice, 0, RTFILE_SEEK_END, pcb);
183}
184
185
186DECLHIDDEN(int) drvHostBaseReadOs(PDRVHOSTBASE pThis, uint64_t off, void *pvBuf, size_t cbRead)
187{
188 return RTFileReadAt(pThis->hFileDevice, off, pvBuf, cbRead, NULL);
189}
190
191
192DECLHIDDEN(int) drvHostBaseWriteOs(PDRVHOSTBASE pThis, uint64_t off, const void *pvBuf, size_t cbWrite)
193{
194 return RTFileWriteAt(pThis->hFileDevice, off, pvBuf, cbWrite, NULL);
195}
196
197
198DECLHIDDEN(int) drvHostBaseFlushOs(PDRVHOSTBASE pThis)
199{
200 return RTFileFlush(pThis->hFileDevice);
201}
202
203
204DECLHIDDEN(int) drvHostBasePollerWakeupOs(PDRVHOSTBASE pThis)
205{
206 return RTSemEventSignal(pThis->EventPoller);
207}
208
209
210DECLHIDDEN(void) drvHostBaseDestructOs(PDRVHOSTBASE pThis)
211{
212 if (pThis->EventPoller != NULL)
213 {
214 RTSemEventDestroy(pThis->EventPoller);
215 pThis->EventPoller = NULL;
216 }
217
218 if (pThis->hFileDevice != NIL_RTFILE)
219 {
220 int rc = RTFileClose(pThis->hFileDevice);
221 AssertRC(rc);
222 pThis->hFileDevice = NIL_RTFILE;
223 }
224
225 if (pThis->hFileRawDevice != NIL_RTFILE)
226 {
227 int rc = RTFileClose(pThis->hFileRawDevice);
228 AssertRC(rc);
229 pThis->hFileRawDevice = NIL_RTFILE;
230 }
231
232 if (pThis->pszRawDeviceOpen)
233 {
234 RTStrFree(pThis->pszRawDeviceOpen);
235 pThis->pszRawDeviceOpen = NULL;
236 }
237}
238
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette