isdn/gigaset: restructure modem response parser (1)
Factor out queueing of modem response events into helper function add_cid_event(). Signed-off-by: Tilman Schmidt <tilman@imap.cc> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
558d51fa2f
commit
6bf5011435
1 changed files with 55 additions and 44 deletions
|
@ -407,6 +407,37 @@ static int cid_of_response(char *s)
|
||||||
return cid;
|
return cid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* queue event with CID */
|
||||||
|
static void add_cid_event(struct cardstate *cs, int cid, int type,
|
||||||
|
void *ptr, int parameter)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
unsigned next, tail;
|
||||||
|
struct event_t *event;
|
||||||
|
|
||||||
|
gig_dbg(DEBUG_EVENT, "queueing event %d for cid %d", type, cid);
|
||||||
|
|
||||||
|
spin_lock_irqsave(&cs->ev_lock, flags);
|
||||||
|
|
||||||
|
tail = cs->ev_tail;
|
||||||
|
next = (tail + 1) % MAX_EVENTS;
|
||||||
|
if (unlikely(next == cs->ev_head)) {
|
||||||
|
dev_err(cs->dev, "event queue full\n");
|
||||||
|
kfree(ptr);
|
||||||
|
} else {
|
||||||
|
event = cs->events + tail;
|
||||||
|
event->type = type;
|
||||||
|
event->cid = cid;
|
||||||
|
event->ptr = ptr;
|
||||||
|
event->arg = NULL;
|
||||||
|
event->parameter = parameter;
|
||||||
|
event->at_state = NULL;
|
||||||
|
cs->ev_tail = next;
|
||||||
|
}
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(&cs->ev_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* gigaset_handle_modem_response() - process received modem response
|
* gigaset_handle_modem_response() - process received modem response
|
||||||
* @cs: device descriptor structure.
|
* @cs: device descriptor structure.
|
||||||
|
@ -420,17 +451,15 @@ void gigaset_handle_modem_response(struct cardstate *cs)
|
||||||
unsigned char *argv[MAX_REC_PARAMS + 1];
|
unsigned char *argv[MAX_REC_PARAMS + 1];
|
||||||
int params;
|
int params;
|
||||||
int i, j;
|
int i, j;
|
||||||
|
char *ptr;
|
||||||
const struct resp_type_t *rt;
|
const struct resp_type_t *rt;
|
||||||
const struct zsau_resp_t *zr;
|
const struct zsau_resp_t *zr;
|
||||||
int curarg;
|
int curarg;
|
||||||
unsigned long flags;
|
|
||||||
unsigned next, tail, head;
|
|
||||||
struct event_t *event;
|
|
||||||
int resp_code;
|
int resp_code;
|
||||||
int param_type;
|
int param_type;
|
||||||
int abort;
|
int abort;
|
||||||
size_t len;
|
size_t len;
|
||||||
int cid;
|
int cid, parameter;
|
||||||
int rawstring;
|
int rawstring;
|
||||||
|
|
||||||
len = cs->cbytes;
|
len = cs->cbytes;
|
||||||
|
@ -484,26 +513,9 @@ void gigaset_handle_modem_response(struct cardstate *cs)
|
||||||
gig_dbg(DEBUG_EVENT, "param %d: %s", j, argv[j]);
|
gig_dbg(DEBUG_EVENT, "param %d: %s", j, argv[j]);
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock_irqsave(&cs->ev_lock, flags);
|
|
||||||
head = cs->ev_head;
|
|
||||||
tail = cs->ev_tail;
|
|
||||||
|
|
||||||
abort = 1;
|
abort = 1;
|
||||||
curarg = 0;
|
curarg = 0;
|
||||||
while (curarg < params) {
|
while (curarg < params) {
|
||||||
next = (tail + 1) % MAX_EVENTS;
|
|
||||||
if (unlikely(next == head)) {
|
|
||||||
dev_err(cs->dev, "event queue full\n");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
event = cs->events + tail;
|
|
||||||
event->at_state = NULL;
|
|
||||||
event->cid = cid;
|
|
||||||
event->ptr = NULL;
|
|
||||||
event->arg = NULL;
|
|
||||||
tail = next;
|
|
||||||
|
|
||||||
if (rawstring) {
|
if (rawstring) {
|
||||||
resp_code = RSP_STRING;
|
resp_code = RSP_STRING;
|
||||||
param_type = RT_STRING;
|
param_type = RT_STRING;
|
||||||
|
@ -513,7 +525,7 @@ void gigaset_handle_modem_response(struct cardstate *cs)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
if (!rt->response) {
|
if (!rt->response) {
|
||||||
event->type = RSP_NONE;
|
add_cid_event(cs, 0, RSP_NONE, NULL, 0);
|
||||||
gig_dbg(DEBUG_EVENT,
|
gig_dbg(DEBUG_EVENT,
|
||||||
"unknown modem response: '%s'\n",
|
"unknown modem response: '%s'\n",
|
||||||
argv[curarg]);
|
argv[curarg]);
|
||||||
|
@ -525,78 +537,77 @@ void gigaset_handle_modem_response(struct cardstate *cs)
|
||||||
++curarg;
|
++curarg;
|
||||||
}
|
}
|
||||||
|
|
||||||
event->type = resp_code;
|
|
||||||
|
|
||||||
switch (param_type) {
|
switch (param_type) {
|
||||||
case RT_NOTHING:
|
case RT_NOTHING:
|
||||||
|
add_cid_event(cs, cid, resp_code, NULL, 0);
|
||||||
break;
|
break;
|
||||||
case RT_RING:
|
case RT_RING:
|
||||||
if (!cid) {
|
if (!cid) {
|
||||||
dev_err(cs->dev,
|
dev_err(cs->dev,
|
||||||
"received RING without CID!\n");
|
"received RING without CID!\n");
|
||||||
event->type = RSP_INVAL;
|
add_cid_event(cs, 0, RSP_INVAL, NULL, 0);
|
||||||
abort = 1;
|
abort = 1;
|
||||||
} else {
|
} else {
|
||||||
event->cid = 0;
|
add_cid_event(cs, 0, resp_code, NULL, cid);
|
||||||
event->parameter = cid;
|
|
||||||
abort = 0;
|
abort = 0;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case RT_ZSAU:
|
case RT_ZSAU:
|
||||||
if (curarg >= params) {
|
if (curarg >= params) {
|
||||||
event->parameter = ZSAU_NONE;
|
add_cid_event(cs, cid, resp_code, NULL,
|
||||||
|
ZSAU_NONE);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
for (zr = zsau_resp; zr->str; ++zr)
|
for (zr = zsau_resp; zr->str; ++zr)
|
||||||
if (!strcmp(argv[curarg], zr->str))
|
if (!strcmp(argv[curarg], zr->str))
|
||||||
break;
|
break;
|
||||||
event->parameter = zr->code;
|
|
||||||
if (!zr->str)
|
if (!zr->str)
|
||||||
dev_warn(cs->dev,
|
dev_warn(cs->dev,
|
||||||
"%s: unknown parameter %s after ZSAU\n",
|
"%s: unknown parameter %s after ZSAU\n",
|
||||||
__func__, argv[curarg]);
|
__func__, argv[curarg]);
|
||||||
|
add_cid_event(cs, cid, resp_code, NULL, zr->code);
|
||||||
++curarg;
|
++curarg;
|
||||||
break;
|
break;
|
||||||
case RT_STRING:
|
case RT_STRING:
|
||||||
if (curarg < params) {
|
if (curarg < params) {
|
||||||
event->ptr = kstrdup(argv[curarg], GFP_ATOMIC);
|
ptr = kstrdup(argv[curarg], GFP_ATOMIC);
|
||||||
if (!event->ptr)
|
if (!ptr)
|
||||||
dev_err(cs->dev, "out of memory\n");
|
dev_err(cs->dev, "out of memory\n");
|
||||||
++curarg;
|
++curarg;
|
||||||
|
} else {
|
||||||
|
ptr = NULL;
|
||||||
}
|
}
|
||||||
gig_dbg(DEBUG_EVENT, "string==%s",
|
gig_dbg(DEBUG_EVENT, "string==%s", ptr ? ptr : "NULL");
|
||||||
event->ptr ? (char *) event->ptr : "NULL");
|
add_cid_event(cs, cid, resp_code, ptr, 0);
|
||||||
break;
|
break;
|
||||||
case RT_ZCAU:
|
case RT_ZCAU:
|
||||||
event->parameter = -1;
|
parameter = -1;
|
||||||
if (curarg + 1 < params) {
|
if (curarg + 1 < params) {
|
||||||
u8 type, value;
|
u8 type, value;
|
||||||
|
|
||||||
i = kstrtou8(argv[curarg++], 16, &type);
|
i = kstrtou8(argv[curarg++], 16, &type);
|
||||||
j = kstrtou8(argv[curarg++], 16, &value);
|
j = kstrtou8(argv[curarg++], 16, &value);
|
||||||
if (i == 0 && j == 0)
|
if (i == 0 && j == 0)
|
||||||
event->parameter = (type << 8) | value;
|
parameter = (type << 8) | value;
|
||||||
} else
|
} else
|
||||||
curarg = params - 1;
|
curarg = params - 1;
|
||||||
|
add_cid_event(cs, cid, resp_code, NULL, parameter);
|
||||||
break;
|
break;
|
||||||
case RT_NUMBER:
|
case RT_NUMBER:
|
||||||
if (curarg >= params ||
|
if (curarg >= params ||
|
||||||
kstrtoint(argv[curarg++], 10, &event->parameter))
|
kstrtoint(argv[curarg++], 10, ¶meter))
|
||||||
event->parameter = -1;
|
parameter = -1;
|
||||||
gig_dbg(DEBUG_EVENT, "parameter==%d", event->parameter);
|
gig_dbg(DEBUG_EVENT, "parameter==%d", parameter);
|
||||||
|
add_cid_event(cs, cid, resp_code, NULL, parameter);
|
||||||
|
if (resp_code == RSP_ZDLE)
|
||||||
|
cs->dle = parameter;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (resp_code == RSP_ZDLE)
|
|
||||||
cs->dle = event->parameter;
|
|
||||||
|
|
||||||
if (abort)
|
if (abort)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
cs->ev_tail = tail;
|
|
||||||
spin_unlock_irqrestore(&cs->ev_lock, flags);
|
|
||||||
|
|
||||||
if (curarg != params)
|
if (curarg != params)
|
||||||
gig_dbg(DEBUG_EVENT,
|
gig_dbg(DEBUG_EVENT,
|
||||||
"invalid number of processed parameters: %d/%d",
|
"invalid number of processed parameters: %d/%d",
|
||||||
|
|
Loading…
Add table
Reference in a new issue