Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/powerpc add clock speed control for 970MP CPUs



details:   https://anonhg.NetBSD.org/src/rev/9b0489deea7d
branches:  trunk
changeset: 323135:9b0489deea7d
user:      macallan <macallan%NetBSD.org@localhost>
date:      Fri Jun 01 18:06:58 2018 +0000

description:
add clock speed control for 970MP CPUs

diffstat:

 sys/arch/powerpc/conf/files.powerpc |    3 +-
 sys/arch/powerpc/oea/cpu_speedctl.c |  183 ++++++++++++++++++++++++++++++++++++
 sys/arch/powerpc/oea/cpu_subr.c     |   12 +-
 3 files changed, 194 insertions(+), 4 deletions(-)

diffs (251 lines):

diff -r 0ad9e0e8cb26 -r 9b0489deea7d sys/arch/powerpc/conf/files.powerpc
--- a/sys/arch/powerpc/conf/files.powerpc       Fri Jun 01 17:18:44 2018 +0000
+++ b/sys/arch/powerpc/conf/files.powerpc       Fri Jun 01 18:06:58 2018 +0000
@@ -1,4 +1,4 @@
-#      $NetBSD: files.powerpc,v 1.89 2014/03/04 08:25:11 matt Exp $
+#      $NetBSD: files.powerpc,v 1.90 2018/06/01 18:06:58 macallan Exp $
 
 defflag        opt_altivec.h   ALTIVEC K_ALTIVEC PPC_HAVE_SPE
 defflag        opt_openpic.h   OPENPIC_DISTRIBUTE
@@ -51,6 +51,7 @@
 defparam opt_oea.h EXTMAP_RANGES=20
 file   arch/powerpc/oea/altivec.c                      (ppc_oea |ppc_oea64 | ppc_oea64_bridge) & altivec
 file   arch/powerpc/oea/cpu_subr.c                     ppc_oea | ppc_oea64 | ppc_oea64_bridge | ppc_oea601
+file   arch/powerpc/oea/cpu_speedctl.c                 ppc_oea64 | ppc_oea64_bridge
 file   arch/powerpc/oea/oea_machdep.c                  ppc_oea | ppc_oea64 | ppc_oea64_bridge | ppc_oea601
 file   arch/powerpc/oea/pmap.c                         ppc_oea | ppc_oea601
 file   arch/powerpc/oea/pmap64.c                       ppc_oea64
diff -r 0ad9e0e8cb26 -r 9b0489deea7d sys/arch/powerpc/oea/cpu_speedctl.c
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sys/arch/powerpc/oea/cpu_speedctl.c       Fri Jun 01 18:06:58 2018 +0000
@@ -0,0 +1,183 @@
+/*     $NetBSD: cpu_speedctl.c,v 1.1 2018/06/01 18:06:58 macallan Exp $ */
+
+/*-
+ * Copyright (c) 2006 Michael Lorenz
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
+ * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
+ * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
+ * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+#include <sys/cdefs.h>
+__KERNEL_RCSID(0, "$NetBSD: cpu_speedctl.c,v 1.1 2018/06/01 18:06:58 macallan Exp $");
+
+#include "opt_ppcparam.h"
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+#include <sys/types.h>
+
+#include <powerpc/psl.h>
+#include <powerpc/spr.h>
+#include <powerpc/oea/cpufeat.h>
+#include <powerpc/oea/spr.h>
+
+#include <sys/sysctl.h>
+#include <dev/ofw/openfirm.h>
+
+void init_scom_speedctl(void);
+static int scom_speedctl = 0;
+static uint32_t scom_speeds[2];
+static uint32_t scom_reg[2];
+static int  sysctl_cpuspeed_temp(SYSCTLFN_ARGS);
+static int  sysctl_cpuspeed_cur(SYSCTLFN_ARGS);
+static int  sysctl_cpuspeed_available(SYSCTLFN_ARGS);
+
+void
+init_scom_speedctl(void)
+{
+       int node;
+       const struct sysctlnode *sysctl_node, *me, *freq;
+
+       /* do this only once */
+       if (scom_speedctl != 0) return;
+       scom_speedctl = 1;
+
+       node = OF_finddevice("/cpus/@0");
+       if (OF_getprop(node, "power-mode-data", scom_reg, 8) != 8)
+               return;
+       if (OF_getprop(node, "clock-frequency", scom_speeds, 4) != 4)
+               return;
+       scom_speeds[0] = scom_speeds[0] / 1000000; /* Hz -> MHz */
+       scom_speeds[1] = scom_speeds[0] / 2;
+
+       sysctl_node = NULL;
+
+       if (sysctl_createv(NULL, 0, NULL, 
+           &me, 
+           CTLFLAG_READWRITE, CTLTYPE_NODE, "cpu", NULL, NULL,
+           0, NULL, 0, CTL_MACHDEP, CTL_CREATE, CTL_EOL) != 0)
+               printf("couldn't create 'cpu' node\n");
+       
+       if (sysctl_createv(NULL, 0, NULL, 
+           &freq, 
+           CTLFLAG_READWRITE, CTLTYPE_NODE, "frequency", NULL, NULL,
+           0, NULL, 0, CTL_MACHDEP, me->sysctl_num, CTL_CREATE, CTL_EOL) != 0)
+               printf("couldn't create 'frequency' node\n");
+
+       if (sysctl_createv(NULL, 0, NULL, 
+           &sysctl_node, 
+           CTLFLAG_READWRITE | CTLFLAG_OWNDESC,
+           CTLTYPE_INT, "target", "CPU speed", sysctl_cpuspeed_temp, 
+           0, NULL, 0, CTL_MACHDEP, me->sysctl_num, freq->sysctl_num, 
+           CTL_CREATE, CTL_EOL) == 0) {
+       } else
+               printf("couldn't create 'target' node\n");
+
+       if (sysctl_createv(NULL, 0, NULL, 
+           &sysctl_node, 
+           CTLFLAG_READWRITE,
+           CTLTYPE_INT, "current", NULL, sysctl_cpuspeed_cur, 
+           1, NULL, 0, CTL_MACHDEP, me->sysctl_num, freq->sysctl_num, 
+           CTL_CREATE, CTL_EOL) == 0) {
+       } else
+               printf("couldn't create 'current' node\n");
+
+       if (sysctl_createv(NULL, 0, NULL, 
+           &sysctl_node, 
+           CTLFLAG_READWRITE,
+           CTLTYPE_STRING, "available", NULL, sysctl_cpuspeed_available, 
+           2, NULL, 0, CTL_MACHDEP, me->sysctl_num, freq->sysctl_num, 
+           CTL_CREATE, CTL_EOL) == 0) {
+       } else
+               printf("couldn't create 'available' node\n");
+}
+
+static int
+sysctl_cpuspeed_temp(SYSCTLFN_ARGS)
+{
+       struct sysctlnode node = *rnode;
+       int speed, nspeed = -1, mhz;
+
+       speed = (scom_read(SCOM_PSR) >> 56) & 3;
+       if (speed > 1) speed = 1;
+       mhz = scom_speeds[speed];
+       node.sysctl_data = &mhz;
+
+       if (sysctl_lookup(SYSCTLFN_CALL(&node)) == 0) {
+               int new_reg;
+
+               new_reg = *(int *)node.sysctl_data;
+               if (new_reg == scom_speeds[0]) {
+                       nspeed = 0;
+               } else if (new_reg == scom_speeds[1]) {
+                       nspeed = 1;
+               } else {
+                       printf("%s: new_reg %d\n", __func__, new_reg);
+                       return EINVAL;
+               }
+               if (nspeed != speed) {
+                       volatile uint64_t junk;
+                       scom_write(SCOM_PCR, 0x80000000);
+                       scom_write(SCOM_PCR, scom_reg[nspeed]);
+                       junk = scom_read(SCOM_PSR);
+                       __USE(junk);
+               }
+               return 0;
+       }
+       return EINVAL;
+}
+
+static int
+sysctl_cpuspeed_cur(SYSCTLFN_ARGS)
+{
+       struct sysctlnode node = *rnode;
+       uint64_t speed;
+       int mhz;
+
+       speed = (scom_read(SCOM_PSR) >> 56) & 3;
+       if (speed > 1) speed = 1;
+
+       mhz = scom_speeds[speed];
+       node.sysctl_data = &mhz;
+       return sysctl_lookup(SYSCTLFN_CALL(&node));
+}
+
+static int
+sysctl_cpuspeed_available(SYSCTLFN_ARGS)
+{
+       struct sysctlnode node = *rnode;
+       char buf[128];
+
+       snprintf(buf, 128, "%d %d", scom_speeds[0], scom_speeds[1]);    
+       node.sysctl_data = buf;
+       return(sysctl_lookup(SYSCTLFN_CALL(&node)));
+}
+
+SYSCTL_SETUP(sysctl_ams_setup, "sysctl obio subtree setup")
+{
+
+       sysctl_createv(NULL, 0, NULL, NULL,
+                      CTLFLAG_PERMANENT,
+                      CTLTYPE_NODE, "machdep", NULL,
+                      NULL, 0, NULL, 0,
+                      CTL_MACHDEP, CTL_EOL);
+}
diff -r 0ad9e0e8cb26 -r 9b0489deea7d sys/arch/powerpc/oea/cpu_subr.c
--- a/sys/arch/powerpc/oea/cpu_subr.c   Fri Jun 01 17:18:44 2018 +0000
+++ b/sys/arch/powerpc/oea/cpu_subr.c   Fri Jun 01 18:06:58 2018 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: cpu_subr.c,v 1.94 2018/05/25 23:00:34 macallan Exp $   */
+/*     $NetBSD: cpu_subr.c,v 1.95 2018/06/01 18:06:58 macallan Exp $   */
 
 /*-
  * Copyright (c) 2001 Matt Thomas.
@@ -34,7 +34,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: cpu_subr.c,v 1.94 2018/05/25 23:00:34 macallan Exp $");
+__KERNEL_RCSID(0, "$NetBSD: cpu_subr.c,v 1.95 2018/06/01 18:06:58 macallan Exp $");
 
 #include "opt_ppcparam.h"
 #include "opt_ppccache.h"
@@ -73,6 +73,8 @@
 static void cpu_tau_refresh(struct sysmon_envsys *, envsys_data_t *);
 #endif
 
+extern void init_scom_speedctl(void);
+
 int cpu = -1;
 int ncpus;
 
@@ -757,6 +759,11 @@
                cpu_tau_setup(ci);
 #endif
 
+#if defined(PPC_OEA64) || defined(PPC_OEA64_BRIDGE)
+       if (vers == IBM970MP)
+               init_scom_speedctl();
+#endif
+
        evcnt_attach_dynamic(&ci->ci_ev_clock, EVCNT_TYPE_INTR,
                NULL, xname, "clock");
        evcnt_attach_dynamic(&ci->ci_ev_traps, EVCNT_TYPE_TRAP,
@@ -1319,7 +1326,6 @@
        h->hatch_hid1 = mfspr(SPR_HID1);
        h->hatch_hid4 = mfspr(SPR_HID4);
        h->hatch_hid5 = mfspr(SPR_HID5);
-       printf("HIDs: %016llx %016llx\n", h->hatch_hid4, h->hatch_hid5);
 #endif
 
        __asm volatile ("mfsdr1 %0" : "=r"(h->hatch_sdr1));



Home | Main Index | Thread Index | Old Index