on 09/03/2010 23:20 vol...@vwsoft.com said the following:
> Hi!
> 
> For some driver enhancements, I need to decide (by code) which
> southbridge (Intel, AMD is all that matters) the driver is facing.
> 
> What's the best (portable wise) way to distinguish the chipset?
> 
> Intel supports two pages of 128 byte CMOS RAM, AMD supports 1 page of
> 256 byte (addressing is different).
> 
> Is there any way to query the CMOS RAM size? I failed to find a way
> while reading Intel & AMD chipset documentation. Older chipsets
> supported only 64 (very old) or 128 byte. Recent (as of for the last 15
> years or so) chipsets supports more.
> 
> As our current nvram(4) driver only works with 128 byte RAM size, is
> anybody interested in seeing the nvram(4) driver enhanced for extended
> memory areas? I do have working code but that assumes an Intel ICH or
> 440LX chipset (fails for SB{67]xx for some reason :).
> 
> Thank you for any pointers!

Volker,

BTW, I have a couple of local patches/hacks for this myself.
But without any auto detection.
Maybe you could find some bits of them useful.

piix4-rtc-nvram-quirk.diff - enables access to upper 128 bytes on PIIX4 (440BX)
systems in chipset configuration

intel-rtc-nvram.diff - enables access to upper 128 bytes, Intel way

dev-nvram.diff - changes behavior of /dev/nvram to what is more convenient for 
me,
breaks compatibility with existing apps using it (if any by now):
1) bytes are counted from offset zero, not 14, but the first 14 bytes (RTC 
status
and control) are always 0xff
2) checksum is not recalculated, it's left for userland to do (not all systems
might want to do that in old BIOS way)

amd-cmos-nvram.diff - enables access to upper 128 bytes, AMD way

Two last patches are on top of the intel-rtc-nvram.diff patch, not against the
clean tree.

-- 
Andriy Gapon
diff --git a/sys/dev/pci/fixup_pci.c b/sys/dev/pci/fixup_pci.c
index 13fc4b1..566e503 100644
--- a/sys/dev/pci/fixup_pci.c
+++ b/sys/dev/pci/fixup_pci.c
@@ -52,6 +52,7 @@ __FBSDID("$FreeBSD$");
 static int     fixup_pci_probe(device_t dev);
 static void    fixwsc_natoma(device_t dev);
 static void    fixc1_nforce2(device_t dev);
+static void    fixrtc_piix4(device_t dev);
 
 static device_method_t fixup_pci_methods[] = {
     /* Device interface */
@@ -77,6 +78,9 @@ fixup_pci_probe(device_t dev)
     case 0x12378086:           /* Intel 82440FX (Natoma) */
        fixwsc_natoma(dev);
        break;
+    case 0x71108086:           /* Intel PIIX4 */
+       fixrtc_piix4(dev);
+       break;
     case 0x01e010de:           /* nVidia nForce2 */
        fixc1_nforce2(dev);
        break;
@@ -105,6 +109,21 @@ fixwsc_natoma(device_t dev)
 #endif
 }
 
+/* Enable access to upper 128-byte bank of RTC NVRAM */
+static void
+fixrtc_piix4(device_t dev)
+{
+    uint8_t            rtccfg;
+
+    rtccfg = pci_read_config(dev, 0xcb, 1);
+    if (!(rtccfg & 0x04)) {
+       printf("Enabling access to RTC NVRAM upper 128-byte extended bank\n");
+       rtccfg |= 0x04;
+       pci_write_config(dev, 0xcb, rtccfg, 1);
+    }
+}
+
+
 /*
  * Set the SYSTEM_IDLE_TIMEOUT to 80 ns on nForce2 systems to work
  * around a hang that is triggered when the CPU generates a very fast
diff --git a/sys/dev/nvram/nvram.c b/sys/dev/nvram/nvram.c
index 164287e..7c323f7 100644
--- a/sys/dev/nvram/nvram.c
+++ b/sys/dev/nvram/nvram.c
@@ -53,7 +53,7 @@
  */
 
 #define NVRAM_FIRST    RTC_DIAG        /* 14 */
-#define NVRAM_LAST     128
+#define NVRAM_LAST     256
 
 #define CKSUM_FIRST    2
 #define CKSUM_LAST     31
@@ -113,6 +113,7 @@ nvram_write(struct cdev *dev, struct uio *uio, int flags)
        u_char v;
        int error = 0;
        int i;
+       int recalc = 0;
        uint16_t sum;
 
        sx_xlock(&nvram_lock);
@@ -129,20 +130,24 @@ nvram_write(struct cdev *dev, struct uio *uio, int flags)
        /* Bring in user data and write */
        while (uio->uio_resid > 0 && error == 0) {
                nv_off = uio->uio_offset + NVRAM_FIRST;
-               if (nv_off < NVRAM_FIRST || nv_off >= NVRAM_LAST) {
-                       sx_xunlock(&nvram_lock);
-                       return (0);     /* Signal EOF */
-               }
+               if (nv_off < NVRAM_FIRST || nv_off >= NVRAM_LAST)
+                       break;
+               if (nv_off >= (NVRAM_FIRST + CKSUM_FIRST)
+                   && nv_off <= (NVRAM_FIRST + CKSUM_LAST))
+                       recalc = 1;
+
                /* Single byte at a time */
                error = uiomove(&v, 1, uio);
                writertc(nv_off, v);
        }
        /* Recalculate checksum afterwards */
-       sum = 0;
-       for (i = CKSUM_FIRST; i <= CKSUM_LAST; i++)
-               sum += rtcin(NVRAM_FIRST + i);
-       writertc(NVRAM_FIRST + CKSUM_MSB, sum >> 8);
-       writertc(NVRAM_FIRST + CKSUM_LSB, sum);
+       if (recalc) {
+               sum = 0;
+               for (i = CKSUM_FIRST; i <= CKSUM_LAST; i++)
+                       sum += rtcin(NVRAM_FIRST + i);
+               writertc(NVRAM_FIRST + CKSUM_MSB, sum >> 8);
+               writertc(NVRAM_FIRST + CKSUM_LSB, sum);
+       }
        sx_xunlock(&nvram_lock);
        return (error);
 }
diff --git a/sys/isa/rtc.h b/sys/isa/rtc.h
index 018a4ed..43a9c15 100644
--- a/sys/isa/rtc.h
+++ b/sys/isa/rtc.h
@@ -115,12 +115,12 @@
 extern  struct mtx clock_lock;
 extern int atrtcclock_disable;
 int    atrtc_setup_clock(void);
-int    rtcin(int reg);
+u_char rtcin(u_char reg);
 void   atrtc_start(void);
 void   atrtc_rate(unsigned rate);
 void   atrtc_enable_intr(void);
 void   atrtc_restore(void);
-void   writertc(int reg, u_char val);
+void   writertc(u_char reg, u_char val);
 #endif
 
 #endif /* _I386_ISA_RTC_H_ */
diff --git a/sys/x86/isa/atrtc.c b/sys/x86/isa/atrtc.c
index 777c720..9f9f0ac 100644
--- a/sys/x86/isa/atrtc.c
+++ b/sys/x86/isa/atrtc.c
@@ -59,35 +59,51 @@ static      u_char  rtc_statusb = RTCSB_24HR;
  * RTC support routines
  */
 
-int
-rtcin(int reg)
+u_char
+rtcin(u_char reg)
 {
        u_char val;
+       int idxr;
+       int datr;
+
+       if (reg & 0x80)
+               idxr = IO_RTC + 2; /* upper bank */
+       else
+               idxr = IO_RTC;     /* lower bank */
+       datr = idxr + 1;
 
        RTC_LOCK;
        if (rtc_reg != reg) {
                inb(0x84);
-               outb(IO_RTC, reg);
+               outb(idxr, reg & 0x7f);
                rtc_reg = reg;
                inb(0x84);
        }
-       val = inb(IO_RTC + 1);
+       val = inb(datr);
        RTC_UNLOCK;
        return (val);
 }
 
 void
-writertc(int reg, u_char val)
+writertc(u_char reg, u_char val)
 {
+       int idxr;
+       int datr;
+
+       if (reg & 0x80)
+               idxr = IO_RTC + 2; /* upper bank */
+       else
+               idxr = IO_RTC;     /* lower bank */
+       datr = idxr + 1;
 
        RTC_LOCK;
        if (rtc_reg != reg) {
                inb(0x84);
-               outb(IO_RTC, reg);
+               outb(idxr, reg & 0x7f);
                rtc_reg = reg;
                inb(0x84);
        }
-       outb(IO_RTC + 1, val);
+       outb(datr, val);
        inb(0x84);
        RTC_UNLOCK;
 }
diff --git a/sys/dev/nvram/nvram.c b/sys/dev/nvram/nvram.c
index 7c323f7..716f4b9 100644
--- a/sys/dev/nvram/nvram.c
+++ b/sys/dev/nvram/nvram.c
@@ -39,33 +39,15 @@
 
 #include <isa/rtc.h>
 
-/*
- * Linux-style /dev/nvram driver
- *
- * cmos ram starts at bytes 14 through 128, for a total of 114 bytes.
- * The driver exposes byte 14 as file offset 0.
- *
- * Offsets 2 through 31 are checksummed at offset 32, 33.
- * In order to avoid the possibility of making the machine unbootable at the
- * bios level (press F1 to continue!), we refuse to allow writes if we do
- * not see a pre-existing valid checksum.  If the existing sum is invalid,
- * then presumably we do not know how to make a sum that the bios will accept.
- */
 
 #define NVRAM_FIRST    RTC_DIAG        /* 14 */
-#define NVRAM_LAST     256
-
-#define CKSUM_FIRST    2
-#define CKSUM_LAST     31
-#define CKSUM_MSB      32
-#define CKSUM_LSB      33
+#define NVRAM_LAST     0xff
 
 static d_open_t                nvram_open;
 static d_read_t                nvram_read;
 static d_write_t       nvram_write;
 
 static struct cdev *nvram_dev;
-static struct sx nvram_lock;
 
 static struct cdevsw nvram_cdevsw = {
        .d_version =    D_VERSION,
@@ -94,12 +76,19 @@ nvram_read(struct cdev *dev, struct uio *uio, int flags)
        u_char v;
        int error = 0;
 
+       if (uio->uio_offset < 0)
+               return 0;
+
        while (uio->uio_resid > 0 && error == 0) {
-               nv_off = uio->uio_offset + NVRAM_FIRST;
-               if (nv_off < NVRAM_FIRST || nv_off >= NVRAM_LAST)
+               nv_off = uio->uio_offset;
+               if (nv_off > NVRAM_LAST)
                        return (0);     /* Signal EOF */
+
                /* Single byte at a time */
-               v = rtcin(nv_off);
+               if (nv_off < NVRAM_FIRST)
+                       v = 0xff;
+               else
+                       v = rtcin(nv_off);
                error = uiomove(&v, 1, uio);
        }
        return (error);
@@ -112,44 +101,28 @@ nvram_write(struct cdev *dev, struct uio *uio, int flags)
        int nv_off;
        u_char v;
        int error = 0;
-       int i;
-       int recalc = 0;
-       uint16_t sum;
-
-       sx_xlock(&nvram_lock);
-
-       /* Assert that we understand the existing checksum first!  */
-       sum = rtcin(NVRAM_FIRST + CKSUM_MSB) << 8 |
-             rtcin(NVRAM_FIRST + CKSUM_LSB);
-       for (i = CKSUM_FIRST; i <= CKSUM_LAST; i++)
-               sum -= rtcin(NVRAM_FIRST + i);
-       if (sum != 0) {
-               sx_xunlock(&nvram_lock);
-               return (EIO);
-       }
+
+       if (uio->uio_offset < 0)
+               return 0;
+
        /* Bring in user data and write */
-       while (uio->uio_resid > 0 && error == 0) {
-               nv_off = uio->uio_offset + NVRAM_FIRST;
-               if (nv_off < NVRAM_FIRST || nv_off >= NVRAM_LAST)
-                       break;
-               if (nv_off >= (NVRAM_FIRST + CKSUM_FIRST)
-                   && nv_off <= (NVRAM_FIRST + CKSUM_LAST))
-                       recalc = 1;
+       while (uio->uio_resid > 0) {
+               nv_off = uio->uio_offset;
+               if (nv_off > NVRAM_LAST)
+                       return (0);
 
                /* Single byte at a time */
                error = uiomove(&v, 1, uio);
+               if (error)
+                       return (error);
+
+               if (nv_off < NVRAM_FIRST)
+                       continue;
+
                writertc(nv_off, v);
        }
-       /* Recalculate checksum afterwards */
-       if (recalc) {
-               sum = 0;
-               for (i = CKSUM_FIRST; i <= CKSUM_LAST; i++)
-                       sum += rtcin(NVRAM_FIRST + i);
-               writertc(NVRAM_FIRST + CKSUM_MSB, sum >> 8);
-               writertc(NVRAM_FIRST + CKSUM_LSB, sum);
-       }
-       sx_xunlock(&nvram_lock);
-       return (error);
+
+       return (0);
 }
 
 static int
@@ -157,14 +130,12 @@ nvram_modevent(module_t mod __unused, int type, void 
*data __unused)
 {
        switch (type) {
        case MOD_LOAD:
-               sx_init(&nvram_lock, "nvram");
                nvram_dev = make_dev(&nvram_cdevsw, 0,
                    UID_ROOT, GID_KMEM, 0640, "nvram");
                break;
        case MOD_UNLOAD:
        case MOD_SHUTDOWN:
                destroy_dev(nvram_dev);
-               sx_destroy(&nvram_lock);
                break;
        default:
                return (EOPNOTSUPP);
diff --git a/sys/amd64/conf/NOTES b/sys/amd64/conf/NOTES
index 8b56e54..d18736e 100644
--- a/sys/amd64/conf/NOTES
+++ b/sys/amd64/conf/NOTES
@@ -86,6 +86,11 @@ options      BPF_JITTER
 # Provide read/write access to the memory in the clock chip.
 device         nvram           # Access to rtc cmos via /dev/nvram
 
+# Whether 72h/73h access to RTC NVRAM should use 7-bit index and
+# access only 128 bytes of upper bank (Intel chipsets) or it can
+# use 8-bit index and access all 256 bytes of NVRAM (AMD chipsets).
+#options       RTC_8BIT_EXTINDEX
+
 
 #####################################################################
 # MISCELLANEOUS DEVICES AND OPTIONS
diff --git a/sys/conf/options.amd64 b/sys/conf/options.amd64
index 5617da4..30af3cc 100644
--- a/sys/conf/options.amd64
+++ b/sys/conf/options.amd64
@@ -61,3 +61,6 @@ KDTRACE_FRAME         opt_kdtrace.h
 BPF_JITTER             opt_bpf.h
 
 XENHVM                 opt_global.h
+
+# ISA options
+RTC_8BIT_EXTINDEX      opt_isa.h
diff --git a/sys/conf/options.i386 b/sys/conf/options.i386
index 83f8286..a4ce060 100644
--- a/sys/conf/options.i386
+++ b/sys/conf/options.i386
@@ -117,3 +117,7 @@ BPF_JITTER          opt_bpf.h
 
 NATIVE                 opt_global.h
 XEN                    opt_global.h
+
+# ISA options
+RTC_8BIT_EXTINDEX      opt_isa.h
+
diff --git a/sys/i386/conf/NOTES b/sys/i386/conf/NOTES
index af3da83..4fb11ae 100644
--- a/sys/i386/conf/NOTES
+++ b/sys/i386/conf/NOTES
@@ -261,6 +261,11 @@ options    BPF_JITTER
 # Provide read/write access to the memory in the clock chip.
 device         nvram           # Access to rtc cmos via /dev/nvram
 
+# Whether 72h/73h access to RTC NVRAM should use 7-bit index and
+# access only 128 bytes of upper bank (Intel chipsets) or it can
+# use 8-bit index and access all 256 bytes of NVRAM (AMD chipsets).
+#options       RTC_8BIT_EXTINDEX
+
 
 #####################################################################
 # MISCELLANEOUS DEVICES AND OPTIONS
diff --git a/sys/x86/isa/atrtc.c b/sys/x86/isa/atrtc.c
index 9f9f0ac..a77e14c 100644
--- a/sys/x86/isa/atrtc.c
+++ b/sys/x86/isa/atrtc.c
@@ -75,7 +75,11 @@ rtcin(u_char reg)
        RTC_LOCK;
        if (rtc_reg != reg) {
                inb(0x84);
+#ifndef RTC_8BIT_EXTINDEX
                outb(idxr, reg & 0x7f);
+#else
+               outb(idxr, reg);
+#endif
                rtc_reg = reg;
                inb(0x84);
        }
@@ -99,7 +103,11 @@ writertc(u_char reg, u_char val)
        RTC_LOCK;
        if (rtc_reg != reg) {
                inb(0x84);
+#ifndef RTC_8BIT_EXTINDEX
                outb(idxr, reg & 0x7f);
+#else
+               outb(idxr, reg);
+#endif
                rtc_reg = reg;
                inb(0x84);
        }
_______________________________________________
freebsd-hackers@freebsd.org mailing list
http://lists.freebsd.org/mailman/listinfo/freebsd-hackers
To unsubscribe, send any mail to "freebsd-hackers-unsubscr...@freebsd.org"

Reply via email to