forked from Imagelibrary/rtems
2001-01-12 Eric Norum <eric.norum@usask.ca>
* README, clock/Makefile.am, clock/ckinit.c, clock/clock.c: Clock driver updated to use shell driver mechanism. * clock/ckinit.c: Removed. * clock/clock.c: New file.
This commit is contained in:
@@ -1,3 +1,10 @@
|
|||||||
|
2001-01-12 Eric Norum <eric.norum@usask.ca>
|
||||||
|
|
||||||
|
* README, clock/Makefile.am, clock/ckinit.c, clock/clock.c:
|
||||||
|
Clock driver updated to use shell driver mechanism.
|
||||||
|
* clock/ckinit.c: Removed.
|
||||||
|
* clock/clock.c: New file.
|
||||||
|
|
||||||
2000-12-19 Joel Sherrill <joel@OARcorp.com>
|
2000-12-19 Joel Sherrill <joel@OARcorp.com>
|
||||||
|
|
||||||
* Makefile.am: Removed unneeded blank line.
|
* Makefile.am: Removed unneeded blank line.
|
||||||
|
|||||||
@@ -302,7 +302,7 @@ The board support package has been tested with:
|
|||||||
|
|
||||||
Interrupt Notes
|
Interrupt Notes
|
||||||
===============
|
===============
|
||||||
ckinit.c:
|
clock.c:
|
||||||
Occasional network lockups have been noted when the PIT has a higher
|
Occasional network lockups have been noted when the PIT has a higher
|
||||||
interrupt request level than the CPM. The SCC1 bit in the CISR is set
|
interrupt request level than the CPM. The SCC1 bit in the CISR is set
|
||||||
even though the SCC1 interrupt handler is not active. This blocks
|
even though the SCC1 interrupt handler is not active. This blocks
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ AUTOMAKE_OPTIONS = foreign 1.4
|
|||||||
|
|
||||||
PGM = $(ARCH)/clock.rel
|
PGM = $(ARCH)/clock.rel
|
||||||
|
|
||||||
C_FILES = ckinit.c
|
C_FILES = clock.c
|
||||||
C_O_FILES = $(C_FILES:%.c=$(ARCH)/%.o)
|
C_O_FILES = $(C_FILES:%.c=$(ARCH)/%.o)
|
||||||
|
|
||||||
OBJS = $(C_O_FILES)
|
OBJS = $(C_O_FILES)
|
||||||
@@ -28,6 +28,6 @@ all-local: $(ARCH) $(OBJS) $(PGM)
|
|||||||
|
|
||||||
.PRECIOUS: $(PGM)
|
.PRECIOUS: $(PGM)
|
||||||
|
|
||||||
EXTRA_DIST = ckinit.c
|
EXTRA_DIST = clock.c
|
||||||
|
|
||||||
include $(top_srcdir)/../../../../../../automake/local.am
|
include $(top_srcdir)/../../../../../../automake/local.am
|
||||||
|
|||||||
@@ -1,192 +0,0 @@
|
|||||||
/*
|
|
||||||
* This routine initializes the MC68360 Periodic Interval Timer
|
|
||||||
*
|
|
||||||
* The PIT has rather poor resolution, but it is easy to set up
|
|
||||||
* and requires no housekeeping once it is going.
|
|
||||||
*
|
|
||||||
* Based on the `gen68302' board support package, and covered by the
|
|
||||||
* original distribution terms.
|
|
||||||
*
|
|
||||||
* W. Eric Norum
|
|
||||||
* Saskatchewan Accelerator Laboratory
|
|
||||||
* University of Saskatchewan
|
|
||||||
* Saskatoon, Saskatchewan, CANADA
|
|
||||||
* eric@skatter.usask.ca
|
|
||||||
*
|
|
||||||
* $Id$
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Input parameters: NONE
|
|
||||||
*
|
|
||||||
* Output parameters: NONE
|
|
||||||
*
|
|
||||||
* COPYRIGHT (c) 1989-1999.
|
|
||||||
* On-Line Applications Research Corporation (OAR).
|
|
||||||
*
|
|
||||||
* The license and distribution terms for this file may be
|
|
||||||
* found in the file LICENSE in this distribution or at
|
|
||||||
* http://www.OARcorp.com/rtems/license.html.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <stdlib.h> /* for atexit() */
|
|
||||||
#include <bsp.h>
|
|
||||||
#include <rtems/libio.h>
|
|
||||||
#include "m68360.h"
|
|
||||||
|
|
||||||
#define CLOCK_VECTOR 120
|
|
||||||
#define CLOCK_IRQ_LEVEL 4
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Clock_driver_ticks is a monotonically increasing counter of the
|
|
||||||
* number of clock ticks since the driver was initialized.
|
|
||||||
*/
|
|
||||||
volatile rtems_unsigned32 Clock_driver_ticks;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* These are set by clock driver during its init
|
|
||||||
*/
|
|
||||||
|
|
||||||
rtems_device_major_number rtems_clock_major = ~0;
|
|
||||||
rtems_device_minor_number rtems_clock_minor;
|
|
||||||
|
|
||||||
char M360DefaultWatchdogFeeder = 1;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* RTEMS and hardware have different notions of clock rate.
|
|
||||||
*/
|
|
||||||
static unsigned long rtems_nsec_per_tick;
|
|
||||||
static unsigned long pit_nsec_per_tick;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Periodic interval timer interrupt handler
|
|
||||||
*/
|
|
||||||
|
|
||||||
rtems_isr
|
|
||||||
Clock_isr (rtems_vector_number vector)
|
|
||||||
{
|
|
||||||
static unsigned long nsec;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* See if it's really time for a `tick'
|
|
||||||
*/
|
|
||||||
nsec += pit_nsec_per_tick;
|
|
||||||
if (nsec >= rtems_nsec_per_tick) {
|
|
||||||
nsec -= rtems_nsec_per_tick;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Perform a dummy read of DPRAM.
|
|
||||||
* This works around a bug in Rev. B of the 68360
|
|
||||||
*/
|
|
||||||
m360.dpram0[0];
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Feed the watchdog
|
|
||||||
* Application code can override this by
|
|
||||||
* setting M360DefaultWatchdogFeeder to zero.
|
|
||||||
*/
|
|
||||||
if (M360DefaultWatchdogFeeder) {
|
|
||||||
m360.swsr = 0x55;
|
|
||||||
m360.swsr = 0xAA;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Announce the clock tick
|
|
||||||
*/
|
|
||||||
Clock_driver_ticks++;
|
|
||||||
rtems_clock_tick();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
Clock_exit (void)
|
|
||||||
{
|
|
||||||
/*
|
|
||||||
* Turn off periodic interval timer
|
|
||||||
*/
|
|
||||||
m360.pitr &= ~0xFF;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
Install_clock (rtems_isr_entry clock_isr)
|
|
||||||
{
|
|
||||||
int divisor;
|
|
||||||
extern int m360_clock_rate; /* This should be somewhere in a config file */
|
|
||||||
unsigned long nsec_per_chip_tick = 1000000000 / m360_clock_rate;
|
|
||||||
unsigned long nsec_per_pit_tick = 512 * nsec_per_chip_tick;
|
|
||||||
|
|
||||||
Clock_driver_ticks = 0;
|
|
||||||
/*
|
|
||||||
* Choose periodic interval timer register value
|
|
||||||
* The rate at which the periodic interval timer
|
|
||||||
* can generate interrupts is almost certainly not
|
|
||||||
* the same as desired by the BSP configuration.
|
|
||||||
* Handle the difference by choosing the largest PIT
|
|
||||||
* interval which is less than or equal to the RTEMS
|
|
||||||
* interval and skipping some hardware interrupts.
|
|
||||||
* To reduce the jitter in the calls to RTEMS the
|
|
||||||
* hardware interrupt interval is never less than
|
|
||||||
* the maximum non-prescaled value from the PIT.
|
|
||||||
*
|
|
||||||
* For a 25 MHz external clock the basic clock rate is
|
|
||||||
* 40 nsec * 128 * 4 = 20.48 usec/tick
|
|
||||||
*/
|
|
||||||
|
|
||||||
rtems_nsec_per_tick = BSP_Configuration.microseconds_per_tick * 1000;
|
|
||||||
divisor = rtems_nsec_per_tick / nsec_per_pit_tick;
|
|
||||||
if (divisor >= 256) {
|
|
||||||
divisor = 255;
|
|
||||||
} else if (divisor == 0) {
|
|
||||||
divisor = 1;
|
|
||||||
}
|
|
||||||
pit_nsec_per_tick = nsec_per_pit_tick * divisor;
|
|
||||||
m360.pitr &= ~0x1FF;
|
|
||||||
m360.picr = (CLOCK_IRQ_LEVEL << 8) | CLOCK_VECTOR;
|
|
||||||
set_vector (clock_isr, CLOCK_VECTOR, 1);
|
|
||||||
m360.pitr |= divisor;
|
|
||||||
atexit (Clock_exit);
|
|
||||||
}
|
|
||||||
|
|
||||||
rtems_device_driver
|
|
||||||
Clock_initialize(
|
|
||||||
rtems_device_major_number major,
|
|
||||||
rtems_device_minor_number minor,
|
|
||||||
void *pargp
|
|
||||||
)
|
|
||||||
{
|
|
||||||
Install_clock (Clock_isr);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* make major/minor avail to others such as shared memory driver
|
|
||||||
*/
|
|
||||||
rtems_clock_major = major;
|
|
||||||
rtems_clock_minor = minor;
|
|
||||||
|
|
||||||
return RTEMS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
|
|
||||||
rtems_device_driver Clock_control(
|
|
||||||
rtems_device_major_number major,
|
|
||||||
rtems_device_minor_number minor,
|
|
||||||
void *pargp
|
|
||||||
)
|
|
||||||
{
|
|
||||||
rtems_unsigned32 isrlevel;
|
|
||||||
rtems_libio_ioctl_args_t *args = pargp;
|
|
||||||
|
|
||||||
if (args) {
|
|
||||||
/*
|
|
||||||
* This is hokey, but until we get a defined interface
|
|
||||||
* to do this, it will just be this simple...
|
|
||||||
*/
|
|
||||||
if (args->command == rtems_build_name('I', 'S', 'R', ' ')) {
|
|
||||||
Clock_isr( CLOCK_VECTOR);
|
|
||||||
}
|
|
||||||
else if (args->command == rtems_build_name('N', 'E', 'W', ' ')) {
|
|
||||||
rtems_interrupt_disable( isrlevel );
|
|
||||||
(void) set_vector( args->buffer, CLOCK_VECTOR, 1 );
|
|
||||||
rtems_interrupt_enable( isrlevel );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return RTEMS_SUCCESSFUL;
|
|
||||||
}
|
|
||||||
102
c/src/lib/libbsp/m68k/gen68360/clock/clock.c
Normal file
102
c/src/lib/libbsp/m68k/gen68360/clock/clock.c
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
/*
|
||||||
|
* This routine initializes the MC68360 Periodic Interval Timer
|
||||||
|
*
|
||||||
|
* The PIT has rather poor resolution, but it is easy to set up
|
||||||
|
* and requires no housekeeping once it is going.
|
||||||
|
*
|
||||||
|
* W. Eric Norum
|
||||||
|
* Saskatchewan Accelerator Laboratory
|
||||||
|
* University of Saskatchewan
|
||||||
|
* Saskatoon, Saskatchewan, CANADA
|
||||||
|
* eric@skatter.usask.ca
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <rtems.h>
|
||||||
|
#include <bsp.h>
|
||||||
|
#include "m68360.h"
|
||||||
|
|
||||||
|
#define CLOCK_VECTOR 120
|
||||||
|
#define CLOCK_IRQ_LEVEL 4
|
||||||
|
|
||||||
|
char M360DefaultWatchdogFeeder = 1;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* RTEMS and hardware have different notions of clock rate.
|
||||||
|
*/
|
||||||
|
static unsigned long rtems_nsec_per_tick;
|
||||||
|
static unsigned long pit_nsec_per_tick;
|
||||||
|
static unsigned long nsec;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Periodic interval timer interrupt handler
|
||||||
|
* See if it's really time for a `tick'
|
||||||
|
* Perform a dummy read of DPRAM (work around bug in Rev. B of the 68360).
|
||||||
|
* Feed the watchdog
|
||||||
|
* Application code can override this by
|
||||||
|
* setting M360DefaultWatchdogFeeder to zero.
|
||||||
|
*/
|
||||||
|
#define Clock_driver_support_at_tick() \
|
||||||
|
do { \
|
||||||
|
nsec += pit_nsec_per_tick; \
|
||||||
|
if (nsec >= rtems_nsec_per_tick) \
|
||||||
|
return; \
|
||||||
|
nsec -= rtems_nsec_per_tick; \
|
||||||
|
m360.dpram0[0]; \
|
||||||
|
if (M360DefaultWatchdogFeeder) { \
|
||||||
|
m360.swsr = 0x55; \
|
||||||
|
m360.swsr = 0xAA; \
|
||||||
|
} \
|
||||||
|
} while (0) \
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Attach clock interrupt handler
|
||||||
|
*/
|
||||||
|
#define Clock_driver_support_install_isr( _new, _old ) \
|
||||||
|
do { \
|
||||||
|
_old = (rtems_isr_entry)set_vector(_new, CLOCK_VECTOR, 1); \
|
||||||
|
} while(0)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Turn off the clock
|
||||||
|
*/
|
||||||
|
#define Clock_driver_support_shutdown_hardware() \
|
||||||
|
do { \
|
||||||
|
m360.pitr &= ~0xFF; \
|
||||||
|
} while(0)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set up the clock hardware
|
||||||
|
* The rate at which the periodic interval timer
|
||||||
|
* can generate interrupts is almost certainly not
|
||||||
|
* the same as desired by the BSP configuration.
|
||||||
|
* Handle the difference by choosing the largest PIT
|
||||||
|
* interval which is less than or equal to the RTEMS
|
||||||
|
* interval and skipping some hardware interrupts.
|
||||||
|
* To reduce the jitter in the calls to RTEMS the
|
||||||
|
* hardware interrupt interval is never greater than
|
||||||
|
* the maximum non-prescaled value from the PIT.
|
||||||
|
*
|
||||||
|
* For a 25 MHz external clock the basic clock rate is
|
||||||
|
* 40 nsec * 128 * 4 = 20.48 usec/tick
|
||||||
|
*/
|
||||||
|
#define Clock_driver_support_initialize_hardware() \
|
||||||
|
do { \
|
||||||
|
unsigned int divisor; \
|
||||||
|
extern int m360_clock_rate; \
|
||||||
|
unsigned long nsec_per_chip_tick = 1000000000 / m360_clock_rate; \
|
||||||
|
unsigned long nsec_per_pit_tick = 512 * nsec_per_chip_tick; \
|
||||||
|
rtems_nsec_per_tick = BSP_Configuration.microseconds_per_tick * 1000; \
|
||||||
|
divisor = rtems_nsec_per_tick / nsec_per_pit_tick; \
|
||||||
|
if (divisor > 255) \
|
||||||
|
divisor = 255; \
|
||||||
|
else if (divisor == 0) \
|
||||||
|
divisor = 1; \
|
||||||
|
pit_nsec_per_tick = nsec_per_pit_tick * divisor; \
|
||||||
|
m360.pitr &= ~0x1FF; \
|
||||||
|
m360.picr = (CLOCK_IRQ_LEVEL << 8) | CLOCK_VECTOR; \
|
||||||
|
m360.pitr |= divisor; \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#include "../../../shared/clockdrv_shell.c"
|
||||||
Reference in New Issue
Block a user