summaryrefslogtreecommitdiffstats
path: root/sys/dev/mfi
diff options
context:
space:
mode:
authorambrisko <ambrisko@FreeBSD.org>2012-11-08 00:32:36 +0000
committerambrisko <ambrisko@FreeBSD.org>2012-11-08 00:32:36 +0000
commita60f06157e6e44ea0691467709b746037e6111c2 (patch)
tree79cbd8e52871b7d172512525b53df1b1200f6ada /sys/dev/mfi
parent9c0bd42e1e1976aa2b56165c1e4e3c5d629e26d2 (diff)
downloadFreeBSD-src-a60f06157e6e44ea0691467709b746037e6111c2.zip
FreeBSD-src-a60f06157e6e44ea0691467709b746037e6111c2.tar.gz
Add support for SCSI pass through devices to be attached and
detached. PR: 172864 Submitted by: rstone@
Diffstat (limited to 'sys/dev/mfi')
-rw-r--r--sys/dev/mfi/mfi.c5
-rw-r--r--sys/dev/mfi/mfi_cam.c71
-rw-r--r--sys/dev/mfi/mfivar.h2
3 files changed, 78 insertions, 0 deletions
diff --git a/sys/dev/mfi/mfi.c b/sys/dev/mfi/mfi.c
index 07b9915..7cb19d2 100644
--- a/sys/dev/mfi/mfi.c
+++ b/sys/dev/mfi/mfi.c
@@ -1577,6 +1577,11 @@ mfi_decode_evt(struct mfi_softc *sc, struct mfi_evt_detail *detail)
sx_xunlock(&sc->mfi_config_lock);
}
}
+ if (sc->mfi_cam_rescan_cb != NULL &&
+ (detail->code == MR_EVT_PD_INSERTED ||
+ detail->code == MR_EVT_PD_REMOVED)) {
+ sc->mfi_cam_rescan_cb(sc, detail->args.pd.device_id);
+ }
break;
}
}
diff --git a/sys/dev/mfi/mfi_cam.c b/sys/dev/mfi/mfi_cam.c
index ce1e415..599ff3b 100644
--- a/sys/dev/mfi/mfi_cam.c
+++ b/sys/dev/mfi/mfi_cam.c
@@ -50,7 +50,9 @@ __FBSDID("$FreeBSD$");
#include <cam/cam.h>
#include <cam/cam_ccb.h>
#include <cam/cam_debug.h>
+#include <cam/cam_periph.h>
#include <cam/cam_sim.h>
+#include <cam/cam_xpt_periph.h>
#include <cam/cam_xpt_sim.h>
#include <cam/scsi/scsi_all.h>
#include <cam/scsi/scsi_message.h>
@@ -63,12 +65,19 @@ __FBSDID("$FreeBSD$");
#include <dev/mfi/mfi_ioctl.h>
#include <dev/mfi/mfivar.h>
+enum mfip_state {
+ MFIP_STATE_NONE,
+ MFIP_STATE_DETACH,
+ MFIP_STATE_RESCAN
+};
+
struct mfip_softc {
device_t dev;
struct mfi_softc *mfi_sc;
struct cam_devq *devq;
struct cam_sim *sim;
struct cam_path *path;
+ enum mfip_state state;
};
static int mfip_probe(device_t);
@@ -76,6 +85,7 @@ static int mfip_attach(device_t);
static int mfip_detach(device_t);
static void mfip_cam_action(struct cam_sim *, union ccb *);
static void mfip_cam_poll(struct cam_sim *);
+static void mfip_cam_rescan(struct mfi_softc *, uint32_t tid);
static struct mfi_command * mfip_start(void *);
static void mfip_done(struct mfi_command *cm);
@@ -122,6 +132,7 @@ mfip_attach(device_t dev)
mfisc = device_get_softc(device_get_parent(dev));
sc->dev = dev;
+ sc->state = MFIP_STATE_NONE;
sc->mfi_sc = mfisc;
mfisc->mfi_cam_start = mfip_start;
@@ -137,6 +148,8 @@ mfip_attach(device_t dev)
return (EINVAL);
}
+ mfisc->mfi_cam_rescan_cb = mfip_cam_rescan;
+
mtx_lock(&mfisc->mfi_io_lock);
if (xpt_bus_register(sc->sim, dev, 0) != 0) {
device_printf(dev, "XPT bus registration failed\n");
@@ -159,6 +172,16 @@ mfip_detach(device_t dev)
if (sc == NULL)
return (EINVAL);
+ mtx_lock(&sc->mfi_sc->mfi_io_lock);
+ if (sc->state == MFIP_STATE_RESCAN) {
+ mtx_unlock(&sc->mfi_sc->mfi_io_lock);
+ return (EBUSY);
+ }
+ sc->state = MFIP_STATE_DETACH;
+ mtx_unlock(&sc->mfi_sc->mfi_io_lock);
+
+ sc->mfi_sc->mfi_cam_rescan_cb = NULL;
+
if (sc->sim != NULL) {
mtx_lock(&sc->mfi_sc->mfi_io_lock);
xpt_bus_deregister(cam_sim_path(sc->sim));
@@ -266,6 +289,54 @@ mfip_cam_action(struct cam_sim *sim, union ccb *ccb)
return;
}
+static void
+mfip_cam_rescan(struct mfi_softc *sc, uint32_t tid)
+{
+ union ccb *ccb;
+ struct mfip_softc *camsc;
+ struct cam_sim *sim;
+ device_t mfip_dev;
+
+ mtx_lock(&Giant);
+ mfip_dev = device_find_child(sc->mfi_dev, "mfip", -1);
+ mtx_unlock(&Giant);
+ if (mfip_dev == NULL) {
+ device_printf(sc->mfi_dev, "Couldn't find mfip child device!\n");
+ return;
+ }
+
+ mtx_lock(&sc->mfi_io_lock);
+ camsc = device_get_softc(mfip_dev);
+ if (camsc->state == MFIP_STATE_DETACH) {
+ mtx_unlock(&sc->mfi_io_lock);
+ return;
+ }
+ camsc->state = MFIP_STATE_RESCAN;
+ mtx_unlock(&sc->mfi_io_lock);
+
+ ccb = xpt_alloc_ccb_nowait();
+ if (ccb == NULL) {
+ device_printf(sc->mfi_dev,
+ "Cannot allocate ccb for bus rescan.\n");
+ return;
+ }
+
+ sim = camsc->sim;
+ if (xpt_create_path(&ccb->ccb_h.path, xpt_periph, cam_sim_path(sim),
+ tid, CAM_LUN_WILDCARD) != CAM_REQ_CMP) {
+ xpt_free_ccb(ccb);
+ device_printf(sc->mfi_dev,
+ "Cannot create path for bus rescan.\n");
+ return;
+ }
+
+ xpt_rescan(ccb);
+
+ mtx_lock(&sc->mfi_io_lock);
+ camsc->state = MFIP_STATE_NONE;
+ mtx_unlock(&sc->mfi_io_lock);
+}
+
static struct mfi_command *
mfip_start(void *data)
{
diff --git a/sys/dev/mfi/mfivar.h b/sys/dev/mfi/mfivar.h
index 46b7c7e..435ca8d 100644
--- a/sys/dev/mfi/mfivar.h
+++ b/sys/dev/mfi/mfivar.h
@@ -303,6 +303,8 @@ struct mfi_softc {
TAILQ_HEAD(, ccb_hdr) mfi_cam_ccbq;
struct mfi_command * (* mfi_cam_start)(void *);
+ void (*mfi_cam_rescan_cb)(struct mfi_softc *,
+ uint32_t);
struct callout mfi_watchdog_callout;
struct mtx mfi_io_lock;
struct sx mfi_config_lock;
OpenPOWER on IntegriCloud