Skip site navigation (1)Skip section navigation (2)
Date:      Mon, 28 May 2007 19:53:19 GMT
From:      Rui Paulo <rpaulo@FreeBSD.org>
To:        Perforce Change Reviews <perforce@FreeBSD.org>
Subject:   PERFORCE change 120510 for review
Message-ID:  <200705281953.l4SJrJPZ073352@repoman.freebsd.org>

next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=120510

Change 120510 by rpaulo@rpaulo_epsilon on 2007/05/28 19:52:46

	Setup the Sudden Motion sensor so that it sends an IRQ when
	the laptop recieves a shock.
	For example, if the laptop is moved in air agressivly, the
	following will show up on dmesg:
	asmc0: WARNING: possible shock!
	asmc0: WARNING: possible shock!
	asmc0: WARNING: high acceleration detected!
	
	Fix comments.

Affected files ...

.. //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmc.c#4 edit
.. //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmcvar.h#3 edit

Differences ...

==== //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmc.c#4 (text+ko) ====

@@ -23,7 +23,7 @@
  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  * POSSIBILITY OF SUCH DAMAGE.
  *
- * $P4: //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmc.c#3 $
+ * $P4: //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmc.c#4 $
  *
  */
 
@@ -76,6 +76,7 @@
 static int	asmc_temp_getvalue(device_t, const char *);
 static int	asmc_sms_read(device_t, const char *, int16_t *);
 static void	asms_sms_calibrate(device_t);
+static int	asmc_sms_intr(void *);
 
 /*
  * Model functions.
@@ -226,6 +227,7 @@
 asmc_attach(device_t dev)
 {
 	int i, j;
+	int error;
 	char name[2];
 	struct asmc_softc *sc = device_get_softc(dev);
 	struct asmc_model *model;
@@ -354,6 +356,26 @@
 			dev, 0, model->smc_sms_z, "I",
 			"Sudden Motion Sensor Z value");
 
+	/*
+	 * Allocate an IRQ for the SMS.
+	 */
+	sc->sc_rid = 0;
+	sc->sc_res = bus_alloc_resource(dev, SYS_RES_IRQ, &sc->sc_rid,
+					ASMC_IRQ, ASMC_IRQ, 1, RF_ACTIVE);
+	if (sc->sc_res == NULL) {
+		device_printf(dev, "unable to allocate IRQ resource\n");
+		goto out;
+	}
+
+	error = bus_setup_intr(dev, sc->sc_res, 
+			       INTR_TYPE_MISC|INTR_MPSAFE|INTR_EXCL,
+			       asmc_sms_intr, NULL, dev, &sc->sc_cookie);
+	if (error) {
+		device_printf(dev, "unable to setup SMS IRQ\n");
+		bus_release_resource(dev, SYS_RES_IRQ, sc->sc_rid,
+				     sc->sc_res);
+	}
+
 out:
 	return 0;
 }
@@ -365,6 +387,11 @@
 
 	sysctl_ctx_free(&sc->sc_sysctl_ctx);
 
+	if (sc->sc_cookie)
+		bus_teardown_intr(dev, sc->sc_res, sc->sc_cookie);
+	if (sc->sc_res)
+		bus_release_resource(dev, SYS_RES_IRQ, sc->sc_rid, sc->sc_res);
+
 	return 0;
 }
 
@@ -376,9 +403,45 @@
 	uint8_t buf[4];
 
 	/*
+	 * We are ready to recieve interrupts from the SMS.
+	 */
+	buf[0] = 0x01;
+	asmc_key_write(dev, ASMC_KEY_INTOK, buf, 1);
+	DELAY(50);
+
+	/* 
+	 * Initiate the polling intervals.
+	 */
+	buf[0] = 20; /* msecs */
+	asmc_key_write(dev, ASMC_KEY_SMS_LOW_INT, buf, 1);
+	DELAY(150);
+
+	buf[0] = 20; /* msecs */
+	asmc_key_write(dev, ASMC_KEY_SMS_HIGH_INT, buf, 1);
+	DELAY(150);
+
+	buf[0] = 0x00;
+	buf[1] = 0x60;
+	asmc_key_write(dev, ASMC_KEY_SMS_LOW, buf, 2);
+	DELAY(150);
+
+	buf[0] = 0x01;
+	buf[1] = 0xc0;
+	asmc_key_write(dev, ASMC_KEY_SMS_HIGH, buf, 2);
+	DELAY(150);
+
+	/*
+	 * I'm not sure what this key does, but it seems to be
+	 * required.
+	 */
+	buf[0] = 0x01;
+	asmc_key_write(dev, ASMC_KEY_SMS_FLAG, buf, 1);
+	DELAY(50);
+
+	/*
 	 * Wait up to 5 seconds for SMS initialization.
 	 */
-	for (i = 0; i < 100; i++) {
+	for (i = 0; i < 10000; i++) {
 		if (asmc_key_read(dev, ASMC_KEY_SMS, buf, 2) == 0 && 
 		    (buf[0] != 0x00 || buf[1] != 0x00)) {
 			error = 0;
@@ -397,7 +460,7 @@
 	sc->sc_nfan = asmc_fan_count(dev);
 	if (sc->sc_nfan > ASMC_MAXFANS) {
 		device_printf(dev, "more than %d fans were detected. Please "
-			           "report this.", ASMC_MAXFANS);
+			           "report this.\n", ASMC_MAXFANS);
 		sc->sc_nfan = ASMC_MAXFANS;
 	}
 
@@ -424,7 +487,7 @@
 
 	val = val & ASMC_STATUS_MASK;
 
-	for (i = 0; i < 10000; i++) {
+	for (i = 0; i < 100; i++) {
 		if ((inb(ASMC_CMDPORT) & ASMC_STATUS_MASK) == val)
 			return 0;
 		DELAY(10);
@@ -668,6 +731,31 @@
 }
 
 static int
+asmc_sms_intr(void *arg)
+{
+	uint8_t type;
+	device_t dev = (device_t) arg;
+
+	type = inb(ASMC_INTPORT);
+
+	switch (type) {
+	case ASMC_SMS_INTFF:
+		device_printf(dev, "WARNING: possible free fall!\n");
+		break;
+	case ASMC_SMS_INTHA:
+		device_printf(dev, "WARNING: high acceleration detected!\n");
+		break;
+	case ASMC_SMS_INTSH:
+		device_printf(dev, "WARNING: possible shock!\n");
+		break;
+	default:
+		device_printf(dev, "%s unknown interrupt\n", __func__);
+	}
+
+	return 0;
+}
+
+static int
 asmc_mb_sysctl_sms_x(SYSCTL_HANDLER_ARGS)
 {
 	device_t dev = (device_t) arg1;

==== //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmcvar.h#3 (text+ko) ====

@@ -23,7 +23,7 @@
  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  * POSSIBILITY OF SUCH DAMAGE.
  *
- * $P4: //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmcvar.h#2 $
+ * $P4: //depot/projects/soc2007/rpaulo-macbook/dev/asmc/asmcvar.h#3 $
  *
  */
 
@@ -45,6 +45,10 @@
 	struct sysctl_oid      *sc_sms_tree;
 
 	struct asmc_model	*sc_model;
+
+	int			sc_rid;
+	struct resource		*sc_res;
+	void			*sc_cookie;
 };
 
 struct asmc_model {
@@ -66,6 +70,13 @@
 	const char	*smc_tempdescs[8];
 };
 
+/*
+ * The Sudden Motion Sensor is able to generate an interrupt when
+ * there are certain critical conditions (free fall, high acceleration and
+ * shocks).
+ * The following IRQ is used.
+ */
+#define ASMC_IRQ		6
 
 /*
  * Data port.
@@ -80,6 +91,12 @@
 #define ASMC_CMDREAD		0x10
 #define ASMC_CMDWRITE		0x11
 
+/*
+ * Interrupt port.
+ */
+#define ASMC_INTPORT		0x31f
+
+
 /* Number of keys */
 #define ASMC_NKEYS		"#KEY"	/* RO; 4 bytes */ 
 
@@ -98,11 +115,19 @@
  * Sudden Motion Sensor (SMS).
  */
 #define ASMC_SMS_INIT1		0xe0
-#define ASMC_SMS_INIT2		0x00
+#define ASMC_SMS_INIT2		0xf8
 #define ASMC_KEY_SMS		"MOCN"	/* RW; 2 bytes */
 #define ASMC_KEY_SMS_X		"MO_X"	/* RO; 2 bytes */
 #define ASMC_KEY_SMS_Y		"MO_Y"	/* RO; 2 bytes */
 #define ASMC_KEY_SMS_Z		"MO_Z"	/* RO; 2 bytes */
+#define ASMC_KEY_SMS_LOW	"MOLT"	/* RW; 2 bytes */
+#define ASMC_KEY_SMS_HIGH	"MOHT"	/* RW; 2 bytes */
+#define ASMC_KEY_SMS_LOW_INT	"MOLD"	/* RW; 1 byte */
+#define ASMC_KEY_SMS_HIGH_INT	"MOHD"	/* RW; 1 byte */
+#define ASMC_KEY_SMS_FLAG	"MSDW"	/* RW; 1 byte */
+#define ASMC_SMS_INTFF		0x60	/* Free fall Interrupt */
+#define ASMC_SMS_INTHA		0x6f	/* High Acceleration Interrupt */
+#define ASMC_SMS_INTSH		0x80	/* Shock Interrupt */
 
 /*
  * Keyboard backlight.
@@ -116,10 +141,16 @@
 #define ASMC_KEY_CLAMSHELL	"MSLD"	/* RO; 1 byte */
 
 /*
+ * Interrupt keys.
+ */
+#define ASMC_KEY_INTOK		"NTOK"	/* WO; 1 byte */
+
+
+
+/*
  * Temperatures.
  *
  * First for MacBook/MacBook Pro, then for Intel Mac Mini.
- * XXX: the descriptions/names are probably wrong (rpaulo)
  * 
  */
 



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200705281953.l4SJrJPZ073352>