1 1.9 dyoung /* $NetBSD: geodecntr.c,v 1.9 2011/07/01 17:37:26 dyoung Exp $ */ 2 1.2 kardel 3 1.2 kardel /*- 4 1.2 kardel * Copyright (c) 2006 The NetBSD Foundation, Inc. 5 1.2 kardel * All rights reserved. 6 1.2 kardel * 7 1.2 kardel * This code is derived from software contributed to The NetBSD Foundation 8 1.2 kardel * by Frank Kardel inspired from the patches to FreeBSD by Poul Henning Kamp 9 1.2 kardel * 10 1.2 kardel * Redistribution and use in source and binary forms, with or without 11 1.2 kardel * modification, are permitted provided that the following conditions 12 1.2 kardel * are met: 13 1.2 kardel * 1. Redistributions of source code must retain the above copyright 14 1.2 kardel * notice, this list of conditions and the following disclaimer. 15 1.2 kardel * 2. Redistributions in binary form must reproduce the above copyright 16 1.2 kardel * notice, this list of conditions and the following disclaimer in the 17 1.2 kardel * documentation and/or other materials provided with the distribution. 18 1.2 kardel * 19 1.2 kardel * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 1.2 kardel * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 1.2 kardel * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 1.2 kardel * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 1.2 kardel * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 1.2 kardel * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 1.2 kardel * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 1.2 kardel * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 1.2 kardel * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 1.2 kardel * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 1.2 kardel * POSSIBILITY OF SUCH DAMAGE. 30 1.2 kardel */ 31 1.2 kardel 32 1.2 kardel #include <sys/cdefs.h> 33 1.9 dyoung __KERNEL_RCSID(0, "$NetBSD: geodecntr.c,v 1.9 2011/07/01 17:37:26 dyoung Exp $"); 34 1.2 kardel 35 1.2 kardel #include <sys/param.h> 36 1.2 kardel #include <sys/systm.h> 37 1.2 kardel #include <sys/kernel.h> 38 1.2 kardel #include <sys/device.h> 39 1.2 kardel #include <machine/cpu.h> 40 1.2 kardel #include <sys/timetc.h> 41 1.9 dyoung #include <sys/bus.h> 42 1.2 kardel #include <arch/i386/pci/geodevar.h> 43 1.2 kardel #include <arch/i386/pci/geodereg.h> 44 1.2 kardel 45 1.2 kardel struct geodecntr_softc { 46 1.2 kardel struct geode_gcb_softc *sc_gcb_dev; 47 1.2 kardel struct timecounter sc_tc; 48 1.2 kardel }; 49 1.2 kardel 50 1.2 kardel static unsigned geode_get_timecount(struct timecounter *); 51 1.2 kardel 52 1.2 kardel static int attached = 0; 53 1.2 kardel 54 1.2 kardel static int 55 1.7 xtraeme geodecntr_match(device_t parent, cfdata_t match, void *aux) 56 1.2 kardel { 57 1.2 kardel return !attached; 58 1.2 kardel } 59 1.2 kardel 60 1.2 kardel /* 61 1.2 kardel * attach time counter 62 1.2 kardel */ 63 1.2 kardel static void 64 1.5 dyoung geodecntr_attach(device_t parent, device_t self, void *aux) 65 1.2 kardel { 66 1.5 dyoung struct geodecntr_softc *sc = device_private(self); 67 1.2 kardel 68 1.8 jmcneill aprint_naive("\n"); 69 1.8 jmcneill aprint_normal(": AMD Geode SC1100 27MHz Counter\n"); 70 1.2 kardel 71 1.5 dyoung sc->sc_gcb_dev = device_private(parent); 72 1.2 kardel 73 1.2 kardel /* 74 1.2 kardel * select 27MHz, no powerdown, no interrupt 75 1.2 kardel */ 76 1.2 kardel bus_space_write_1(sc->sc_gcb_dev->sc_iot, sc->sc_gcb_dev->sc_ioh, 77 1.2 kardel SC1100_GCB_TMCNFG_B, SC1100_TMCNFG_TMCLKSEL); 78 1.2 kardel 79 1.2 kardel memset(&sc->sc_tc, 0, sizeof(sc->sc_tc)); 80 1.2 kardel 81 1.2 kardel sc->sc_tc.tc_get_timecount = geode_get_timecount; 82 1.2 kardel sc->sc_tc.tc_poll_pps = NULL; 83 1.2 kardel sc->sc_tc.tc_counter_mask = 0xffffffff; 84 1.2 kardel sc->sc_tc.tc_frequency = 27000000; 85 1.2 kardel sc->sc_tc.tc_name = "geodecounter"; 86 1.2 kardel sc->sc_tc.tc_priv = sc; 87 1.2 kardel sc->sc_tc.tc_quality = 1000; 88 1.2 kardel 89 1.2 kardel tc_init(&sc->sc_tc); 90 1.2 kardel 91 1.2 kardel attached = 1; 92 1.2 kardel } 93 1.2 kardel 94 1.5 dyoung static int 95 1.5 dyoung geodecntr_detach(device_t self, int flags) 96 1.5 dyoung { 97 1.5 dyoung struct geodecntr_softc *sc = device_private(self); 98 1.5 dyoung 99 1.5 dyoung attached = 0; 100 1.5 dyoung return tc_detach(&sc->sc_tc); 101 1.5 dyoung } 102 1.5 dyoung 103 1.2 kardel /* 104 1.2 kardel * read counter 105 1.2 kardel */ 106 1.2 kardel static unsigned geode_get_timecount(struct timecounter *tc) 107 1.2 kardel { 108 1.2 kardel struct geodecntr_softc *sc = (struct geodecntr_softc *)tc->tc_priv; 109 1.2 kardel 110 1.5 dyoung return bus_space_read_4(sc->sc_gcb_dev->sc_iot, sc->sc_gcb_dev->sc_ioh, 111 1.5 dyoung SC1100_GCB_TMVALUE_L); 112 1.2 kardel } 113 1.2 kardel 114 1.7 xtraeme CFATTACH_DECL_NEW(geodecntr, sizeof(struct geodecntr_softc), 115 1.5 dyoung geodecntr_match, geodecntr_attach, geodecntr_detach, NULL); 116 1.2 kardel 117