Use kpause(9), cv_timedwait(9), and cv_signal(9) instead of tsleep(9)
and wakeup(9). Use mstohz(9). XXX Protection against spurious wakeups is still needed, but this patch XXX makes the code no worse than before in this regard.
This commit is contained in:
parent
076d7e3e4e
commit
487ea78a9c
@ -1,4 +1,4 @@
|
|||||||
/* $NetBSD: pccbb.c,v 1.188 2009/05/21 17:32:32 dyoung Exp $ */
|
/* $NetBSD: pccbb.c,v 1.189 2009/07/23 21:22:25 dyoung Exp $ */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 1998, 1999 and 2000
|
* Copyright (c) 1998, 1999 and 2000
|
||||||
@ -31,7 +31,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <sys/cdefs.h>
|
#include <sys/cdefs.h>
|
||||||
__KERNEL_RCSID(0, "$NetBSD: pccbb.c,v 1.188 2009/05/21 17:32:32 dyoung Exp $");
|
__KERNEL_RCSID(0, "$NetBSD: pccbb.c,v 1.189 2009/07/23 21:22:25 dyoung Exp $");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define CBB_DEBUG
|
#define CBB_DEBUG
|
||||||
@ -89,12 +89,12 @@ int pccbb_burstup = 1;
|
|||||||
* of delay() if you want to wait more than 1 ms.
|
* of delay() if you want to wait more than 1 ms.
|
||||||
*/
|
*/
|
||||||
static inline void
|
static inline void
|
||||||
delay_ms(int millis, void *param)
|
delay_ms(int millis, struct pccbb_softc *sc)
|
||||||
{
|
{
|
||||||
if (cold)
|
if (cold)
|
||||||
delay(millis * 1000);
|
delay(millis * 1000);
|
||||||
else
|
else
|
||||||
tsleep(param, PWAIT, "pccbb", MAX(2, hz * millis / 1000));
|
kpause("pccbb", false, mstohz(millis), NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
int pcicbbmatch(device_t, cfdata_t, void *);
|
int pcicbbmatch(device_t, cfdata_t, void *);
|
||||||
@ -422,6 +422,9 @@ pccbbattach(device_t parent, device_t self, void *aux)
|
|||||||
|
|
||||||
sc->sc_dev = self;
|
sc->sc_dev = self;
|
||||||
|
|
||||||
|
mutex_init(&sc->sc_pwr_mtx, MUTEX_DEFAULT, IPL_BIO);
|
||||||
|
cv_init(&sc->sc_pwr_cv, "pccpwr");
|
||||||
|
|
||||||
callout_init(&sc->sc_insert_ch, 0);
|
callout_init(&sc->sc_insert_ch, 0);
|
||||||
callout_setfunc(&sc->sc_insert_ch, pci113x_insert, sc);
|
callout_setfunc(&sc->sc_insert_ch, pci113x_insert, sc);
|
||||||
|
|
||||||
@ -1069,8 +1072,10 @@ pccbbintr(void *arg)
|
|||||||
if (sockevent & CB_SOCKET_EVENT_POWER) {
|
if (sockevent & CB_SOCKET_EVENT_POWER) {
|
||||||
DPRINTF(("Powercycling because of socket event\n"));
|
DPRINTF(("Powercycling because of socket event\n"));
|
||||||
/* XXX: Does not happen when attaching a 16-bit card */
|
/* XXX: Does not happen when attaching a 16-bit card */
|
||||||
|
mutex_enter(&sc->sc_pwr_mtx);
|
||||||
sc->sc_pwrcycle++;
|
sc->sc_pwrcycle++;
|
||||||
wakeup(&sc->sc_pwrcycle);
|
cv_signal(&sc->sc_pwr_cv);
|
||||||
|
mutex_exit(&sc->sc_pwr_mtx);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Sometimes a change of CSTSCHG# accompanies the first
|
/* Sometimes a change of CSTSCHG# accompanies the first
|
||||||
@ -1129,7 +1134,7 @@ pccbbintr(void *arg)
|
|||||||
if (sc->sc_flags & CBB_INSERTING) {
|
if (sc->sc_flags & CBB_INSERTING) {
|
||||||
callout_stop(&sc->sc_insert_ch);
|
callout_stop(&sc->sc_insert_ch);
|
||||||
}
|
}
|
||||||
callout_schedule(&sc->sc_insert_ch, hz / 5);
|
callout_schedule(&sc->sc_insert_ch, mstohz(200));
|
||||||
sc->sc_flags |= CBB_INSERTING;
|
sc->sc_flags |= CBB_INSERTING;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1197,7 +1202,7 @@ pci113x_insert(void *arg)
|
|||||||
/* who are you? */
|
/* who are you? */
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
callout_schedule(&sc->sc_insert_ch, hz / 10);
|
callout_schedule(&sc->sc_insert_ch, mstohz(100));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1285,7 +1290,7 @@ pccbb_power(struct pccbb_softc *sc, int command)
|
|||||||
u_int32_t status, osock_ctrl, sock_ctrl, reg_ctrl;
|
u_int32_t status, osock_ctrl, sock_ctrl, reg_ctrl;
|
||||||
bus_space_tag_t memt = sc->sc_base_memt;
|
bus_space_tag_t memt = sc->sc_base_memt;
|
||||||
bus_space_handle_t memh = sc->sc_base_memh;
|
bus_space_handle_t memh = sc->sc_base_memh;
|
||||||
int on = 0, pwrcycle, s, times;
|
int on = 0, pwrcycle, times;
|
||||||
struct timeval before, after, diff;
|
struct timeval before, after, diff;
|
||||||
|
|
||||||
DPRINTF(("pccbb_power: %s and %s [0x%x]\n",
|
DPRINTF(("pccbb_power: %s and %s [0x%x]\n",
|
||||||
@ -1352,13 +1357,13 @@ pccbb_power(struct pccbb_softc *sc, int command)
|
|||||||
sock_ctrl |= CB_SOCKET_CTRL_VPP_12V;
|
sock_ctrl |= CB_SOCKET_CTRL_VPP_12V;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
pwrcycle = sc->sc_pwrcycle;
|
|
||||||
aprint_debug_dev(sc->sc_dev, "osock_ctrl %#" PRIx32
|
aprint_debug_dev(sc->sc_dev, "osock_ctrl %#" PRIx32
|
||||||
" sock_ctrl %#" PRIx32 "\n", osock_ctrl, sock_ctrl);
|
" sock_ctrl %#" PRIx32 "\n", osock_ctrl, sock_ctrl);
|
||||||
|
|
||||||
microtime(&before);
|
microtime(&before);
|
||||||
s = splbio();
|
mutex_enter(&sc->sc_pwr_mtx);
|
||||||
|
pwrcycle = sc->sc_pwrcycle;
|
||||||
|
|
||||||
bus_space_write_4(memt, memh, CB_SOCKET_CTRL, sock_ctrl);
|
bus_space_write_4(memt, memh, CB_SOCKET_CTRL, sock_ctrl);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1380,8 +1385,8 @@ pccbb_power(struct pccbb_softc *sc, int command)
|
|||||||
if (cold)
|
if (cold)
|
||||||
DELAY(40 * 1000);
|
DELAY(40 * 1000);
|
||||||
else {
|
else {
|
||||||
(void)tsleep(&sc->sc_pwrcycle, PWAIT, "pccpwr",
|
(void)cv_timedwait(&sc->sc_pwr_cv, &sc->sc_pwr_mtx,
|
||||||
hz / 25);
|
mstohz(40));
|
||||||
if (pwrcycle == sc->sc_pwrcycle)
|
if (pwrcycle == sc->sc_pwrcycle)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -1391,7 +1396,7 @@ pccbb_power(struct pccbb_softc *sc, int command)
|
|||||||
if ((status & CB_SOCKET_STAT_PWRCYCLE) == 0 && !on)
|
if ((status & CB_SOCKET_STAT_PWRCYCLE) == 0 && !on)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
splx(s);
|
mutex_exit(&sc->sc_pwr_mtx);
|
||||||
microtime(&after);
|
microtime(&after);
|
||||||
timersub(&after, &before, &diff);
|
timersub(&after, &before, &diff);
|
||||||
aprint_debug_dev(sc->sc_dev, "wait took%s %lld.%06lds\n",
|
aprint_debug_dev(sc->sc_dev, "wait took%s %lld.%06lds\n",
|
||||||
@ -2307,7 +2312,7 @@ pccbb_pcmcia_delay(struct pccbb_softc *sc, int timo, const char *wmesg)
|
|||||||
panic("pccbb_pcmcia_delay: called in interrupt context");
|
panic("pccbb_pcmcia_delay: called in interrupt context");
|
||||||
#endif
|
#endif
|
||||||
DPRINTF(("pccbb_pcmcia_delay: \"%s\", sleep %d ms\n", wmesg, timo));
|
DPRINTF(("pccbb_pcmcia_delay: \"%s\", sleep %d ms\n", wmesg, timo));
|
||||||
tsleep(pccbb_pcmcia_delay, PWAIT, wmesg, roundup(timo * hz, 1000) / 1000);
|
kpause(wmesg, false, max(mstohz(timo), 1), NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* $NetBSD: pccbbvar.h,v 1.37 2008/06/26 20:57:10 drochner Exp $ */
|
/* $NetBSD: pccbbvar.h,v 1.38 2009/07/23 21:22:25 dyoung Exp $ */
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 1999 HAYAKAWA Koichi. All rights reserved.
|
* Copyright (c) 1999 HAYAKAWA Koichi. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -36,6 +36,9 @@
|
|||||||
#ifndef _DEV_PCI_PCCBBVAR_H_
|
#ifndef _DEV_PCI_PCCBBVAR_H_
|
||||||
#define _DEV_PCI_PCCBBVAR_H_
|
#define _DEV_PCI_PCCBBVAR_H_
|
||||||
|
|
||||||
|
#include <sys/mutex.h>
|
||||||
|
#include <sys/condvar.h>
|
||||||
|
|
||||||
#define PCIC_FLAG_SOCKETP 0x0001
|
#define PCIC_FLAG_SOCKETP 0x0001
|
||||||
#define PCIC_FLAG_CARDP 0x0002
|
#define PCIC_FLAG_CARDP 0x0002
|
||||||
|
|
||||||
@ -133,6 +136,8 @@ struct pccbb_softc {
|
|||||||
#define PCCBB_PCMCIA_MEM_32 0x02 /* 32-bit memory address ready */
|
#define PCCBB_PCMCIA_MEM_32 0x02 /* 32-bit memory address ready */
|
||||||
|
|
||||||
volatile int sc_pwrcycle;
|
volatile int sc_pwrcycle;
|
||||||
|
kcondvar_t sc_pwr_cv;
|
||||||
|
kmutex_t sc_pwr_mtx;
|
||||||
|
|
||||||
/* interrupt handler list on the bridge */
|
/* interrupt handler list on the bridge */
|
||||||
LIST_HEAD(, pccbb_intrhand_list) sc_pil;
|
LIST_HEAD(, pccbb_intrhand_list) sc_pil;
|
||||||
|
Loading…
Reference in New Issue
Block a user