mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-23 22:25:40 +08:00
1f7c14afd4
In case the serial port or cable got faulty, we may not be getting acknowledgements any more. The driver then currently waits for 4s to avoid jamming the device. This makes this delay configurable. Signed-off-by: Samuel Thibault <samuel.thibault@ens-lyon.org> Link: https://lore.kernel.org/r/20210128180116.1848120-3-samuel.thibault@ens-lyon.org Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
341 lines
8.1 KiB
C
341 lines
8.1 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
#include <linux/ctype.h>
|
|
#include "spk_types.h"
|
|
#include "spk_priv.h"
|
|
#include "speakup.h"
|
|
|
|
static struct st_var_header var_headers[] = {
|
|
{ "version", VERSION, VAR_PROC, NULL, NULL },
|
|
{ "synth_name", SYNTH, VAR_PROC, NULL, NULL },
|
|
{ "keymap", KEYMAP, VAR_PROC, NULL, NULL },
|
|
{ "silent", SILENT, VAR_PROC, NULL, NULL },
|
|
{ "punc_some", PUNC_SOME, VAR_PROC, NULL, NULL },
|
|
{ "punc_most", PUNC_MOST, VAR_PROC, NULL, NULL },
|
|
{ "punc_all", PUNC_ALL, VAR_PROC, NULL, NULL },
|
|
{ "delimiters", DELIM, VAR_PROC, NULL, NULL },
|
|
{ "repeats", REPEATS, VAR_PROC, NULL, NULL },
|
|
{ "ex_num", EXNUMBER, VAR_PROC, NULL, NULL },
|
|
{ "characters", CHARS, VAR_PROC, NULL, NULL },
|
|
{ "synth_direct", SYNTH_DIRECT, VAR_PROC, NULL, NULL },
|
|
{ "caps_start", CAPS_START, VAR_STRING, spk_str_caps_start, NULL },
|
|
{ "caps_stop", CAPS_STOP, VAR_STRING, spk_str_caps_stop, NULL },
|
|
{ "delay_time", DELAY, VAR_TIME, NULL, NULL },
|
|
{ "trigger_time", TRIGGER, VAR_TIME, NULL, NULL },
|
|
{ "jiffy_delta", JIFFY, VAR_TIME, NULL, NULL },
|
|
{ "full_time", FULL, VAR_TIME, NULL, NULL },
|
|
{ "flush_time", FLUSH, VAR_TIME, NULL, NULL },
|
|
{ "spell_delay", SPELL_DELAY, VAR_NUM, &spk_spell_delay, NULL },
|
|
{ "bleeps", BLEEPS, VAR_NUM, &spk_bleeps, NULL },
|
|
{ "attrib_bleep", ATTRIB_BLEEP, VAR_NUM, &spk_attrib_bleep, NULL },
|
|
{ "bleep_time", BLEEP_TIME, VAR_TIME, &spk_bleep_time, NULL },
|
|
{ "cursor_time", CURSOR_TIME, VAR_TIME, NULL, NULL },
|
|
{ "punc_level", PUNC_LEVEL, VAR_NUM, &spk_punc_level, NULL },
|
|
{ "reading_punc", READING_PUNC, VAR_NUM, &spk_reading_punc, NULL },
|
|
{ "say_control", SAY_CONTROL, VAR_NUM, &spk_say_ctrl, NULL },
|
|
{ "say_word_ctl", SAY_WORD_CTL, VAR_NUM, &spk_say_word_ctl, NULL },
|
|
{ "no_interrupt", NO_INTERRUPT, VAR_NUM, &spk_no_intr, NULL },
|
|
{ "key_echo", KEY_ECHO, VAR_NUM, &spk_key_echo, NULL },
|
|
{ "bell_pos", BELL_POS, VAR_NUM, &spk_bell_pos, NULL },
|
|
{ "rate", RATE, VAR_NUM, NULL, NULL },
|
|
{ "pitch", PITCH, VAR_NUM, NULL, NULL },
|
|
{ "inflection", INFLECTION, VAR_NUM, NULL, NULL },
|
|
{ "vol", VOL, VAR_NUM, NULL, NULL },
|
|
{ "tone", TONE, VAR_NUM, NULL, NULL },
|
|
{ "punct", PUNCT, VAR_NUM, NULL, NULL },
|
|
{ "voice", VOICE, VAR_NUM, NULL, NULL },
|
|
{ "freq", FREQUENCY, VAR_NUM, NULL, NULL },
|
|
{ "lang", LANG, VAR_NUM, NULL, NULL },
|
|
{ "chartab", CHARTAB, VAR_PROC, NULL, NULL },
|
|
{ "direct", DIRECT, VAR_NUM, NULL, NULL },
|
|
{ "pause", PAUSE, VAR_STRING, spk_str_pause, NULL },
|
|
};
|
|
|
|
static struct st_var_header *var_ptrs[MAXVARS] = { NULL, NULL, NULL };
|
|
|
|
static struct punc_var_t punc_vars[] = {
|
|
{ PUNC_SOME, 1 },
|
|
{ PUNC_MOST, 2 },
|
|
{ PUNC_ALL, 3 },
|
|
{ DELIM, 4 },
|
|
{ REPEATS, 5 },
|
|
{ EXNUMBER, 6 },
|
|
{ -1, -1 },
|
|
};
|
|
|
|
int spk_chartab_get_value(char *keyword)
|
|
{
|
|
int value = 0;
|
|
|
|
if (!strcmp(keyword, "ALPHA"))
|
|
value = ALPHA;
|
|
else if (!strcmp(keyword, "B_CTL"))
|
|
value = B_CTL;
|
|
else if (!strcmp(keyword, "WDLM"))
|
|
value = WDLM;
|
|
else if (!strcmp(keyword, "A_PUNC"))
|
|
value = A_PUNC;
|
|
else if (!strcmp(keyword, "PUNC"))
|
|
value = PUNC;
|
|
else if (!strcmp(keyword, "NUM"))
|
|
value = NUM;
|
|
else if (!strcmp(keyword, "A_CAP"))
|
|
value = A_CAP;
|
|
else if (!strcmp(keyword, "B_CAPSYM"))
|
|
value = B_CAPSYM;
|
|
else if (!strcmp(keyword, "B_SYM"))
|
|
value = B_SYM;
|
|
return value;
|
|
}
|
|
|
|
void speakup_register_var(struct var_t *var)
|
|
{
|
|
static char nothing[2] = "\0";
|
|
int i;
|
|
struct st_var_header *p_header;
|
|
|
|
BUG_ON(!var || var->var_id < 0 || var->var_id >= MAXVARS);
|
|
if (!var_ptrs[0]) {
|
|
for (i = 0; i < MAXVARS; i++) {
|
|
p_header = &var_headers[i];
|
|
var_ptrs[p_header->var_id] = p_header;
|
|
p_header->data = NULL;
|
|
}
|
|
}
|
|
p_header = var_ptrs[var->var_id];
|
|
if (p_header->data)
|
|
return;
|
|
p_header->data = var;
|
|
switch (p_header->var_type) {
|
|
case VAR_STRING:
|
|
spk_set_string_var(nothing, p_header, 0);
|
|
break;
|
|
case VAR_NUM:
|
|
case VAR_TIME:
|
|
spk_set_num_var(0, p_header, E_DEFAULT);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void speakup_unregister_var(enum var_id_t var_id)
|
|
{
|
|
struct st_var_header *p_header;
|
|
|
|
BUG_ON(var_id < 0 || var_id >= MAXVARS);
|
|
p_header = var_ptrs[var_id];
|
|
p_header->data = NULL;
|
|
}
|
|
|
|
struct st_var_header *spk_get_var_header(enum var_id_t var_id)
|
|
{
|
|
struct st_var_header *p_header;
|
|
|
|
if (var_id < 0 || var_id >= MAXVARS)
|
|
return NULL;
|
|
p_header = var_ptrs[var_id];
|
|
if (!p_header->data)
|
|
return NULL;
|
|
return p_header;
|
|
}
|
|
|
|
struct st_var_header *spk_var_header_by_name(const char *name)
|
|
{
|
|
int i;
|
|
|
|
if (!name)
|
|
return NULL;
|
|
|
|
for (i = 0; i < MAXVARS; i++) {
|
|
if (strcmp(name, var_ptrs[i]->name) == 0)
|
|
return var_ptrs[i];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
struct var_t *spk_get_var(enum var_id_t var_id)
|
|
{
|
|
BUG_ON(var_id < 0 || var_id >= MAXVARS);
|
|
BUG_ON(!var_ptrs[var_id]);
|
|
return var_ptrs[var_id]->data;
|
|
}
|
|
EXPORT_SYMBOL_GPL(spk_get_var);
|
|
|
|
struct punc_var_t *spk_get_punc_var(enum var_id_t var_id)
|
|
{
|
|
struct punc_var_t *rv = NULL;
|
|
struct punc_var_t *where;
|
|
|
|
where = punc_vars;
|
|
while ((where->var_id != -1) && (!rv)) {
|
|
if (where->var_id == var_id)
|
|
rv = where;
|
|
else
|
|
where++;
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
/* handlers for setting vars */
|
|
int spk_set_num_var(int input, struct st_var_header *var, int how)
|
|
{
|
|
int val;
|
|
int *p_val = var->p_val;
|
|
char buf[32];
|
|
char *cp;
|
|
struct var_t *var_data = var->data;
|
|
|
|
if (!var_data)
|
|
return -ENODATA;
|
|
|
|
val = var_data->u.n.value;
|
|
switch (how) {
|
|
case E_NEW_DEFAULT:
|
|
if (input < var_data->u.n.low || input > var_data->u.n.high)
|
|
return -ERANGE;
|
|
var_data->u.n.default_val = input;
|
|
return 0;
|
|
case E_DEFAULT:
|
|
val = var_data->u.n.default_val;
|
|
break;
|
|
case E_SET:
|
|
val = input;
|
|
break;
|
|
case E_INC:
|
|
val += input;
|
|
break;
|
|
case E_DEC:
|
|
val -= input;
|
|
break;
|
|
}
|
|
|
|
if (val < var_data->u.n.low || val > var_data->u.n.high)
|
|
return -ERANGE;
|
|
|
|
var_data->u.n.value = val;
|
|
if (var->var_type == VAR_TIME && p_val) {
|
|
*p_val = msecs_to_jiffies(val);
|
|
return 0;
|
|
}
|
|
if (p_val)
|
|
*p_val = val;
|
|
if (var->var_id == PUNC_LEVEL) {
|
|
spk_punc_mask = spk_punc_masks[val];
|
|
return 0;
|
|
}
|
|
if (var_data->u.n.multiplier != 0)
|
|
val *= var_data->u.n.multiplier;
|
|
val += var_data->u.n.offset;
|
|
if (var->var_id < FIRST_SYNTH_VAR || !synth)
|
|
return 0;
|
|
if (synth->synth_adjust)
|
|
return synth->synth_adjust(var);
|
|
|
|
if (!var_data->u.n.synth_fmt)
|
|
return 0;
|
|
if (var->var_id == PITCH)
|
|
cp = spk_pitch_buff;
|
|
else
|
|
cp = buf;
|
|
if (!var_data->u.n.out_str)
|
|
sprintf(cp, var_data->u.n.synth_fmt, (int)val);
|
|
else
|
|
sprintf(cp, var_data->u.n.synth_fmt,
|
|
var_data->u.n.out_str[val]);
|
|
synth_printf("%s", cp);
|
|
return 0;
|
|
}
|
|
|
|
int spk_set_string_var(const char *page, struct st_var_header *var, int len)
|
|
{
|
|
struct var_t *var_data = var->data;
|
|
|
|
if (!var_data)
|
|
return -ENODATA;
|
|
if (len > MAXVARLEN)
|
|
return -E2BIG;
|
|
if (!len) {
|
|
if (!var_data->u.s.default_val)
|
|
return 0;
|
|
if (!var->p_val)
|
|
var->p_val = var_data->u.s.default_val;
|
|
if (var->p_val != var_data->u.s.default_val)
|
|
strcpy((char *)var->p_val, var_data->u.s.default_val);
|
|
return -ERESTART;
|
|
} else if (var->p_val) {
|
|
strcpy((char *)var->p_val, page);
|
|
} else {
|
|
return -E2BIG;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* spk_set_mask_bits sets or clears the punc/delim/repeat bits,
|
|
* if input is null uses the defaults.
|
|
* values for how: 0 clears bits of chars supplied,
|
|
* 1 clears allk, 2 sets bits for chars
|
|
*/
|
|
int spk_set_mask_bits(const char *input, const int which, const int how)
|
|
{
|
|
u_char *cp;
|
|
short mask = spk_punc_info[which].mask;
|
|
|
|
if (how & 1) {
|
|
for (cp = (u_char *)spk_punc_info[3].value; *cp; cp++)
|
|
spk_chartab[*cp] &= ~mask;
|
|
}
|
|
cp = (u_char *)input;
|
|
if (!cp) {
|
|
cp = spk_punc_info[which].value;
|
|
} else {
|
|
for (; *cp; cp++) {
|
|
if (*cp < SPACE)
|
|
break;
|
|
if (mask < PUNC) {
|
|
if (!(spk_chartab[*cp] & PUNC))
|
|
break;
|
|
} else if (spk_chartab[*cp] & B_NUM) {
|
|
break;
|
|
}
|
|
}
|
|
if (*cp)
|
|
return -EINVAL;
|
|
cp = (u_char *)input;
|
|
}
|
|
if (how & 2) {
|
|
for (; *cp; cp++)
|
|
if (*cp > SPACE)
|
|
spk_chartab[*cp] |= mask;
|
|
} else {
|
|
for (; *cp; cp++)
|
|
if (*cp > SPACE)
|
|
spk_chartab[*cp] &= ~mask;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
char *spk_strlwr(char *s)
|
|
{
|
|
char *p;
|
|
|
|
if (!s)
|
|
return NULL;
|
|
|
|
for (p = s; *p; p++)
|
|
*p = tolower(*p);
|
|
return s;
|
|
}
|
|
|
|
char *spk_s2uchar(char *start, char *dest)
|
|
{
|
|
int val;
|
|
|
|
/* Do not replace with kstrtoul: here we need start to be updated */
|
|
val = simple_strtoul(skip_spaces(start), &start, 10);
|
|
if (*start == ',')
|
|
start++;
|
|
*dest = (u_char)val;
|
|
return start;
|
|
}
|