Re: [PATCH] Add support for Possio GCC AKA PCMCIA Siemens MC45

From: Petr Vandrovec
Date: Mon Aug 30 2004 - 06:13:31 EST


On Mon, Aug 30, 2004 at 01:05:46AM -0700, Andrew Morton wrote:
> Petr Vandrovec <vandrove@xxxxxxxxxx> wrote:
> >
> > This ugly hack add support for Siemens MC45 PCMCIA GPRS card (which is
> > identical to Possio GCC, and which is offered by one of our local GPRS
> > providers).
>
> drivers/serial/serial_cs.c: In function `wakeup_card':
> drivers/serial/serial_cs.c:139: `MANFID_SIEMENS' undeclared (first use in this function)
> drivers/serial/serial_cs.c:139: (Each undeclared identifier is reported only once
> drivers/serial/serial_cs.c:139: for each function it appears in.)
> drivers/serial/serial_cs.c:139: `PRODID_SIEMENS_MC45' undeclared (first use in this function)
> drivers/serial/serial_cs.c: In function `multi_config':
> drivers/serial/serial_cs.c:594: `MANFID_SIEMENS' undeclared (first use in this function)
> drivers/serial/serial_cs.c:594: `PRODID_SIEMENS_MC45' undeclared (first use in this function)

Oops, I forgot to include linux/include/pcmcia/ciscode.h patch.
Please revert old one (with SIEMENS) and apply one below.

I also found that 030C is not SIEMENS but System Innovation
AB - and System Innovation AB is a company within Possio
group. And as PRODID_SYSTEM_INOVATION_GCC was too long, and
product is sold under Possio GCC name, I decided to use
PRODID_POSSIO_GCC.



Hello,

This ugly hack add support for Siemens MC45 PCMCIA GPRS card (which is
identical to Possio GCC, and which is offered by one of our local GPRS
providers). Card has unfortunate feature that after poweron oxcf950 chip
is fully powered and works, but attached MC45 modem is powered down :-(
There is a special sequence (which takes 1 sec :-( ) to poweron
MC45 (and after MC45 powers on, it takes more than 2 secs until firmware
fully boots...) which needs to be executed after all powerons.

I'm really not familiar with PCMCIA subsystem, so I have no idea whether
I should issue request_region() on rest of oxcf950 address range (0-7
is UART, 8-F are special configuration registers), or how this should be
better integrated with PM system and so on - I just put it in same place
where another hack already lived...

Card uses 18.432MHz XTAL, so to get it to work you must add lines below
to the /etc/pcmcia/serial.opts.

case "$MANFID-$FUNCID-$PRODID_1-$PRODID_2-$PRODID_3-$PRODID_4" in
'030c,0003-2-GPRS-CARD--')
SERIAL_OPTS="baud_base 1152000"
;;
esac

Best regards,
Petr Vandrovec



diff -urN linux/include/pcmcia/ciscode.h linux/include/pcmcia/ciscode.h
--- linux/include/pcmcia/ciscode.h 2004-08-28 23:21:03.000000000 +0200
+++ linux/include/pcmcia/ciscode.h 2004-08-28 23:36:24.000000000 +0200
@@ -135,4 +135,7 @@

#define MANFID_XIRCOM 0x0105

+#define MANFID_POSSIO 0x030c
+#define PRODID_POSSIO_GCC 0x0003
+
#endif /* _LINUX_CISCODE_H */
diff -urN linux/drivers/serial/serial_cs.c linux/drivers/serial/serial_cs.c
--- linux/drivers/serial/serial_cs.c 2004-08-28 23:19:58.000000000 +0200
+++ linux/drivers/serial/serial_cs.c 2004-08-30 02:53:05.000000000 +0200
@@ -44,6 +44,7 @@
#include <linux/serial.h>
#include <linux/serial_core.h>
#include <linux/major.h>
+#include <linux/delay.h>
#include <asm/io.h>
#include <asm/system.h>

@@ -112,6 +113,8 @@
int multi;
int slave;
int manfid;
+ int prodid;
+ int c950ctrl;
dev_node_t node[4];
int line[4];
};
@@ -127,6 +130,33 @@

static dev_link_t *dev_list = NULL;

+static void wakeup_card(struct serial_info *info)
+{
+ int ctrl = info->c950ctrl;
+
+ if (info->manfid == MANFID_OXSEMI) {
+ outb(12, ctrl + 1);
+ } else if (info->manfid == MANFID_POSSIO && info->prodid == PRODID_POSSIO_GCC) {
+ /* request_region? oxsemi branch does no request_region too... */
+ /* This sequence is needed to properly initialize MC45 attached to OXCF950.
+ * I tried decreasing these msleep()s, but it worked properly (survived
+ * 1000 stop/start operations) with these timeouts (or bigger). */
+ outb(0xA, ctrl + 1);
+ msleep(100);
+ outb(0xE, ctrl + 1);
+ msleep(300);
+ outb(0xC, ctrl + 1);
+ msleep(100);
+ outb(0xE, ctrl + 1);
+ msleep(200);
+ outb(0xF, ctrl + 1);
+ msleep(100);
+ outb(0xE, ctrl + 1);
+ msleep(100);
+ outb(0xC, ctrl + 1);
+ }
+}
+
/*======================================================================

After a card is removed, serial_remove() will unregister
@@ -191,6 +221,7 @@

for (i = 0; i < info->ndev; i++)
serial8250_resume_port(info->line[i]);
+ wakeup_card(info);
}
}

@@ -554,15 +585,20 @@
}

/* The Oxford Semiconductor OXCF950 cards are in fact single-port:
- 8 registers are for the UART, the others are extra registers */
- if (info->manfid == MANFID_OXSEMI) {
+ 8 registers are for the UART, the others are extra registers.
+ Siemen's MC45 PCMCIA (Possio's GCC) is OXCF950 based too. */
+ if (info->manfid == MANFID_OXSEMI ||
+ (info->manfid == MANFID_POSSIO && info->prodid == PRODID_POSSIO_GCC)) {
+ int err;
+
if (cf->index == 1 || cf->index == 3) {
- setup_serial(info, base2, link->irq.AssignedIRQ);
- outb(12, link->io.BasePort1 + 1);
+ err = setup_serial(info, base2, link->irq.AssignedIRQ);
+ base2 = link->io.BasePort1;
} else {
- setup_serial(info, link->io.BasePort1, link->irq.AssignedIRQ);
- outb(12, base2 + 1);
+ err = setup_serial(info, link->io.BasePort1, link->irq.AssignedIRQ);
}
+ info->c950ctrl = base2;
+ wakeup_card(info);
return 0;
}

@@ -622,9 +658,10 @@
tuple.DesiredTuple = CISTPL_MANFID;
if (first_tuple(handle, &tuple, &parse) == CS_SUCCESS) {
info->manfid = le16_to_cpu(buf[0]);
+ info->prodid = le16_to_cpu(buf[1]);
for (i = 0; i < MULTI_COUNT; i++)
if ((info->manfid == multi_id[i].manfid) &&
- (le16_to_cpu(buf[1]) == multi_id[i].prodid))
+ (info->prodid == multi_id[i].prodid))
break;
if (i < MULTI_COUNT)
info->multi = multi_id[i].multi;
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/