Linux Audio

Check our new training course

Real-Time Linux with PREEMPT_RT training

Feb 18-20, 2025
Register
Loading...
v3.1
 
  1#include <linux/slab.h> /* for kmalloc */
  2#include <linux/consolemap.h>
  3#include <linux/interrupt.h>
  4#include <linux/sched.h>
 
  5#include <linux/selection.h>
 
 
 
 
 
  6
  7#include "speakup.h"
  8
  9/* ------ cut and paste ----- */
 10/* Don't take this from <ctype.h>: 011-015 on the screen aren't spaces */
 11#define ishardspace(c)      ((c) == ' ')
 12
 13unsigned short xs, ys, xe, ye; /* our region points */
 14
 15/* Variables for selection control. */
 16/* must not be disallocated */
 17struct vc_data *spk_sel_cons;
 18/* cleared by clear_selection */
 19static int sel_start = -1;
 20static int sel_end;
 21static int sel_buffer_lth;
 22static char *sel_buffer;
 23
 24static unsigned char sel_pos(int n)
 25{
 26	return inverse_translate(spk_sel_cons,
 27		screen_glyph(spk_sel_cons, n), 0);
 28}
 29
 30void speakup_clear_selection(void)
 31{
 32	sel_start = -1;
 
 
 33}
 34
 35/* does screen address p correspond to character at LH/RH edge of screen? */
 36static int atedge(const int p, int size_row)
 37{
 38	return !(p % size_row) || !((p + 2) % size_row);
 39}
 40
 41/* constrain v such that v <= u */
 42static unsigned short limit(const unsigned short v, const unsigned short u)
 43{
 44	return (v > u) ? u : v;
 45}
 46
 47int speakup_set_selection(struct tty_struct *tty)
 48{
 49	int new_sel_start, new_sel_end;
 50	char *bp, *obp;
 51	int i, ps, pe;
 52	struct vc_data *vc = vc_cons[fg_console].d;
 53
 54	xs = limit(xs, vc->vc_cols - 1);
 55	ys = limit(ys, vc->vc_rows - 1);
 56	xe = limit(xe, vc->vc_cols - 1);
 57	ye = limit(ye, vc->vc_rows - 1);
 58	ps = ys * vc->vc_size_row + (xs << 1);
 59	pe = ye * vc->vc_size_row + (xe << 1);
 60
 61	if (ps > pe) {
 62		/* make sel_start <= sel_end */
 63		int tmp = ps;
 64		ps = pe;
 65		pe = tmp;
 66	}
 67
 68	if (spk_sel_cons != vc_cons[fg_console].d) {
 69		speakup_clear_selection();
 70		spk_sel_cons = vc_cons[fg_console].d;
 71		printk(KERN_WARNING
 72			"Selection: mark console not the same as cut\n");
 73		return -EINVAL;
 74	}
 75
 76	new_sel_start = ps;
 77	new_sel_end = pe;
 
 78
 79	/* select to end of line if on trailing space */
 80	if (new_sel_end > new_sel_start &&
 81	    !atedge(new_sel_end, vc->vc_size_row) &&
 82	    ishardspace(sel_pos(new_sel_end))) {
 83		for (pe = new_sel_end + 2; ; pe += 2)
 84			if (!ishardspace(sel_pos(pe)) ||
 85			    atedge(pe, vc->vc_size_row))
 86				break;
 87		if (ishardspace(sel_pos(pe)))
 88			new_sel_end = pe;
 89	}
 90	if ((new_sel_start == sel_start) && (new_sel_end == sel_end))
 91		return 0; /* no action required */
 92
 93	sel_start = new_sel_start;
 94	sel_end = new_sel_end;
 95	/* Allocate a new buffer before freeing the old one ... */
 96	bp = kmalloc((sel_end-sel_start)/2+1, GFP_ATOMIC);
 97	if (!bp) {
 98		printk(KERN_WARNING "selection: kmalloc() failed\n");
 99		speakup_clear_selection();
100		return -ENOMEM;
101	}
102	kfree(sel_buffer);
103	sel_buffer = bp;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
104
105	obp = bp;
106	for (i = sel_start; i <= sel_end; i += 2) {
107		*bp = sel_pos(i);
108		if (!ishardspace(*bp++))
109			obp = bp;
110		if (!((i + 2) % vc->vc_size_row)) {
111			/* strip trailing blanks from line and add newline,
112			   unless non-space at end of line. */
113			if (obp != bp) {
114				bp = obp;
115				*bp++ = '\r';
116			}
117			obp = bp;
118		}
119	}
120	sel_buffer_lth = bp - sel_buffer;
121	return 0;
122}
123
124/* TODO: move to some helper thread, probably.  That'd fix having to check for
125 * in_atomic().  */
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
126int speakup_paste_selection(struct tty_struct *tty)
127{
128	struct vc_data *vc = (struct vc_data *) tty->driver_data;
129	int pasted = 0, count;
130	DECLARE_WAITQUEUE(wait, current);
131	add_wait_queue(&vc->paste_wait, &wait);
132	while (sel_buffer && sel_buffer_lth > pasted) {
133		set_current_state(TASK_INTERRUPTIBLE);
134		if (test_bit(TTY_THROTTLED, &tty->flags)) {
135			if (in_atomic())
136				/* if we are in an interrupt handler, abort */
137				break;
138			schedule();
139			continue;
140		}
141		count = sel_buffer_lth - pasted;
142		count = min_t(int, count, tty->receive_room);
143		tty->ldisc->ops->receive_buf(tty, sel_buffer + pasted,
144			0, count);
145		pasted += count;
146	}
147	remove_wait_queue(&vc->paste_wait, &wait);
148	current->state = TASK_RUNNING;
149	return 0;
150}
151
v5.4
  1// SPDX-License-Identifier: GPL-2.0
  2#include <linux/slab.h> /* for kmalloc */
  3#include <linux/consolemap.h>
  4#include <linux/interrupt.h>
  5#include <linux/sched.h>
  6#include <linux/device.h> /* for dev_warn */
  7#include <linux/selection.h>
  8#include <linux/workqueue.h>
  9#include <linux/tty.h>
 10#include <linux/tty_flip.h>
 11#include <linux/atomic.h>
 12#include <linux/console.h>
 13
 14#include "speakup.h"
 15
 16unsigned short spk_xs, spk_ys, spk_xe, spk_ye; /* our region points */
 
 
 
 
 
 
 
 17struct vc_data *spk_sel_cons;
 
 
 
 
 
 18
 19struct speakup_selection_work {
 20	struct work_struct work;
 21	struct tiocl_selection sel;
 22	struct tty_struct *tty;
 23};
 24
 25void speakup_clear_selection(void)
 26{
 27	console_lock();
 28	clear_selection();
 29	console_unlock();
 30}
 31
 32static void __speakup_set_selection(struct work_struct *work)
 
 33{
 34	struct speakup_selection_work *ssw =
 35		container_of(work, struct speakup_selection_work, work);
 36
 37	struct tty_struct *tty;
 38	struct tiocl_selection sel;
 
 
 
 39
 40	sel = ssw->sel;
 41
 42	/* this ensures we copy sel before releasing the lock below */
 43	rmb();
 44
 45	/* release the lock by setting tty of the struct to NULL */
 46	tty = xchg(&ssw->tty, NULL);
 
 
 
 
 
 
 
 
 
 
 
 
 
 47
 48	if (spk_sel_cons != vc_cons[fg_console].d) {
 
 49		spk_sel_cons = vc_cons[fg_console].d;
 50		pr_warn("Selection: mark console not the same as cut\n");
 51		goto unref;
 
 52	}
 53
 54	console_lock();
 55	set_selection_kernel(&sel, tty);
 56	console_unlock();
 57
 58unref:
 59	tty_kref_put(tty);
 60}
 
 
 
 
 
 
 
 
 
 
 61
 62static struct speakup_selection_work speakup_sel_work = {
 63	.work = __WORK_INITIALIZER(speakup_sel_work.work,
 64				   __speakup_set_selection)
 65};
 66
 67int speakup_set_selection(struct tty_struct *tty)
 68{
 69	/* we get kref here first in order to avoid a subtle race when
 70	 * cancelling selection work. getting kref first establishes the
 71	 * invariant that if speakup_sel_work.tty is not NULL when
 72	 * speakup_cancel_selection() is called, it must be the case that a put
 73	 * kref is pending.
 74	 */
 75	tty_kref_get(tty);
 76	if (cmpxchg(&speakup_sel_work.tty, NULL, tty)) {
 77		tty_kref_put(tty);
 78		return -EBUSY;
 79	}
 80	/* now we have the 'lock' by setting tty member of
 81	 * speakup_selection_work. wmb() ensures that writes to
 82	 * speakup_sel_work don't happen before cmpxchg() above.
 83	 */
 84	wmb();
 85
 86	speakup_sel_work.sel.xs = spk_xs + 1;
 87	speakup_sel_work.sel.ys = spk_ys + 1;
 88	speakup_sel_work.sel.xe = spk_xe + 1;
 89	speakup_sel_work.sel.ye = spk_ye + 1;
 90	speakup_sel_work.sel.sel_mode = TIOCL_SELCHAR;
 91
 92	schedule_work_on(WORK_CPU_UNBOUND, &speakup_sel_work.work);
 93
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 94	return 0;
 95}
 96
 97void speakup_cancel_selection(void)
 98{
 99	struct tty_struct *tty;
100
101	cancel_work_sync(&speakup_sel_work.work);
102	/* setting to null so that if work fails to run and we cancel it,
103	 * we can run it again without getting EBUSY forever from there on.
104	 * we need to use xchg here to avoid race with speakup_set_selection()
105	 */
106	tty = xchg(&speakup_sel_work.tty, NULL);
107	if (tty)
108		tty_kref_put(tty);
109}
110
111static void __speakup_paste_selection(struct work_struct *work)
112{
113	struct speakup_selection_work *ssw =
114		container_of(work, struct speakup_selection_work, work);
115	struct tty_struct *tty = xchg(&ssw->tty, NULL);
116
117	paste_selection(tty);
118	tty_kref_put(tty);
119}
120
121static struct speakup_selection_work speakup_paste_work = {
122	.work = __WORK_INITIALIZER(speakup_paste_work.work,
123				   __speakup_paste_selection)
124};
125
126int speakup_paste_selection(struct tty_struct *tty)
127{
128	tty_kref_get(tty);
129	if (cmpxchg(&speakup_paste_work.tty, NULL, tty)) {
130		tty_kref_put(tty);
131		return -EBUSY;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
132	}
133
134	schedule_work_on(WORK_CPU_UNBOUND, &speakup_paste_work.work);
135	return 0;
136}
137
138void speakup_cancel_paste(void)
139{
140	struct tty_struct *tty;
141
142	cancel_work_sync(&speakup_paste_work.work);
143	tty = xchg(&speakup_paste_work.tty, NULL);
144	if (tty)
145		tty_kref_put(tty);
146}