mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 2067fd92d7
			
		
	
	
		2067fd92d7
		
	
	
	
	
		
			
			The nasty TODO items are done. Signed-off-by: Samuel Thibault <samuel.thibault@ens-lyon.org> Link: https://lore.kernel.org/r/20200729003531.907370-1-samuel.thibault@ens-lyon.org Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
		
			
				
	
	
		
			125 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			125 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| #include <linux/console.h>
 | |
| #include <linux/types.h>
 | |
| #include <linux/wait.h>
 | |
| 
 | |
| #include "speakup.h"
 | |
| #include "spk_priv.h"
 | |
| 
 | |
| #define SYNTH_BUF_SIZE 8192	/* currently 8K bytes */
 | |
| 
 | |
| static u16 synth_buffer[SYNTH_BUF_SIZE];	/* guess what this is for! */
 | |
| static u16 *buff_in = synth_buffer;
 | |
| static u16 *buff_out = synth_buffer;
 | |
| static u16 *buffer_end = synth_buffer + SYNTH_BUF_SIZE - 1;
 | |
| 
 | |
| /* These try to throttle applications by stopping the TTYs
 | |
|  * Note: we need to make sure that we will restart them eventually, which is
 | |
|  * usually not possible to do from the notifiers. TODO: it should be possible
 | |
|  * starting from linux 2.6.26.
 | |
|  *
 | |
|  * So we only stop when we know alive == 1 (else we discard the data anyway),
 | |
|  * and the alive synth will eventually call start_ttys from the thread context.
 | |
|  */
 | |
| void speakup_start_ttys(void)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	for (i = 0; i < MAX_NR_CONSOLES; i++) {
 | |
| 		if (speakup_console[i] && speakup_console[i]->tty_stopped)
 | |
| 			continue;
 | |
| 		if (vc_cons[i].d && vc_cons[i].d->port.tty)
 | |
| 			start_tty(vc_cons[i].d->port.tty);
 | |
| 	}
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(speakup_start_ttys);
 | |
| 
 | |
| static void speakup_stop_ttys(void)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	for (i = 0; i < MAX_NR_CONSOLES; i++)
 | |
| 		if (vc_cons[i].d && vc_cons[i].d->port.tty)
 | |
| 			stop_tty(vc_cons[i].d->port.tty);
 | |
| }
 | |
| 
 | |
| static int synth_buffer_free(void)
 | |
| {
 | |
| 	int chars_free;
 | |
| 
 | |
| 	if (buff_in >= buff_out)
 | |
| 		chars_free = SYNTH_BUF_SIZE - (buff_in - buff_out);
 | |
| 	else
 | |
| 		chars_free = buff_out - buff_in;
 | |
| 	return chars_free;
 | |
| }
 | |
| 
 | |
| int synth_buffer_empty(void)
 | |
| {
 | |
| 	return (buff_in == buff_out);
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(synth_buffer_empty);
 | |
| 
 | |
| void synth_buffer_add(u16 ch)
 | |
| {
 | |
| 	if (!synth->alive) {
 | |
| 		/* This makes sure that we won't stop TTYs if there is no synth
 | |
| 		 * to restart them
 | |
| 		 */
 | |
| 		return;
 | |
| 	}
 | |
| 	if (synth_buffer_free() <= 100) {
 | |
| 		synth_start();
 | |
| 		speakup_stop_ttys();
 | |
| 	}
 | |
| 	if (synth_buffer_free() <= 1)
 | |
| 		return;
 | |
| 	*buff_in++ = ch;
 | |
| 	if (buff_in > buffer_end)
 | |
| 		buff_in = synth_buffer;
 | |
| 	/* We have written something to the speech synthesis, so we are not
 | |
| 	 * paused any more.
 | |
| 	 */
 | |
| 	spk_paused = false;
 | |
| }
 | |
| 
 | |
| u16 synth_buffer_getc(void)
 | |
| {
 | |
| 	u16 ch;
 | |
| 
 | |
| 	if (buff_out == buff_in)
 | |
| 		return 0;
 | |
| 	ch = *buff_out++;
 | |
| 	if (buff_out > buffer_end)
 | |
| 		buff_out = synth_buffer;
 | |
| 	return ch;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(synth_buffer_getc);
 | |
| 
 | |
| u16 synth_buffer_peek(void)
 | |
| {
 | |
| 	if (buff_out == buff_in)
 | |
| 		return 0;
 | |
| 	return *buff_out;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(synth_buffer_peek);
 | |
| 
 | |
| void synth_buffer_skip_nonlatin1(void)
 | |
| {
 | |
| 	while (buff_out != buff_in) {
 | |
| 		if (*buff_out < 0x100)
 | |
| 			return;
 | |
| 		buff_out++;
 | |
| 		if (buff_out > buffer_end)
 | |
| 			buff_out = synth_buffer;
 | |
| 	}
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(synth_buffer_skip_nonlatin1);
 | |
| 
 | |
| void synth_buffer_clear(void)
 | |
| {
 | |
| 	buff_in = synth_buffer;
 | |
| 	buff_out = synth_buffer;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(synth_buffer_clear);
 |