cregit-Linux how code gets into the kernel

Release 4.11 drivers/staging/speakup/selection.c

#include <linux/slab.h> /* for kmalloc */
#include <linux/consolemap.h>
#include <linux/interrupt.h>
#include <linux/sched.h>
#include <linux/device.h> /* for dev_warn */
#include <linux/selection.h>
#include <linux/workqueue.h>
#include <linux/tty.h>
#include <linux/tty_flip.h>
#include <linux/atomic.h>

#include "speakup.h"

/* ------ cut and paste ----- */
/* Don't take this from <ctype.h>: 011-015 on the screen aren't spaces */

#define ishardspace(c)      ((c) == ' ')





unsigned short spk_xs, spk_ys, spk_xe, spk_ye; 



/* our region points */

/* Variables for selection control. */
/* must not be deallocated */

struct vc_data *spk_sel_cons;
/* cleared by clear_selection */

static int sel_start = -1;

static int sel_end;

static int sel_buffer_lth;

static char *sel_buffer;


static unsigned char sel_pos(int n) { return inverse_translate(spk_sel_cons, screen_glyph(spk_sel_cons, n), 0); }

Contributors

PersonTokensPropCommitsCommitProp
William Hubbs25100.00%1100.00%
Total25100.00%1100.00%


void speakup_clear_selection(void) { sel_start = -1; }

Contributors

PersonTokensPropCommitsCommitProp
William Hubbs12100.00%1100.00%
Total12100.00%1100.00%

/* does screen address p correspond to character at LH/RH edge of screen? */
static int atedge(const int p, int size_row) { return !(p % size_row) || !((p + 2) % size_row); }

Contributors

PersonTokensPropCommitsCommitProp
William Hubbs32100.00%1100.00%
Total32100.00%1100.00%

/* constrain v such that v <= u */
static unsigned short limit(const unsigned short v, const unsigned short u) { return (v > u) ? u : v; }

Contributors

PersonTokensPropCommitsCommitProp
William Hubbs28100.00%1100.00%
Total28100.00%1100.00%


int speakup_set_selection(struct tty_struct *tty) { int new_sel_start, new_sel_end; char *bp, *obp; int i, ps, pe; struct vc_data *vc = vc_cons[fg_console].d; spk_xs = limit(spk_xs, vc->vc_cols - 1); spk_ys = limit(spk_ys, vc->vc_rows - 1); spk_xe = limit(spk_xe, vc->vc_cols - 1); spk_ye = limit(spk_ye, vc->vc_rows - 1); ps = spk_ys * vc->vc_size_row + (spk_xs << 1); pe = spk_ye * vc->vc_size_row + (spk_xe << 1); if (ps > pe) { /* make sel_start <= sel_end */ int tmp = ps; ps = pe; pe = tmp; } if (spk_sel_cons != vc_cons[fg_console].d) { speakup_clear_selection(); spk_sel_cons = vc_cons[fg_console].d; dev_warn(tty->dev, "Selection: mark console not the same as cut\n"); return -EINVAL; } new_sel_start = ps; new_sel_end = pe; /* select to end of line if on trailing space */ if (new_sel_end > new_sel_start && !atedge(new_sel_end, vc->vc_size_row) && ishardspace(sel_pos(new_sel_end))) { for (pe = new_sel_end + 2; ; pe += 2) if (!ishardspace(sel_pos(pe)) || atedge(pe, vc->vc_size_row)) break; if (ishardspace(sel_pos(pe))) new_sel_end = pe; } if ((new_sel_start == sel_start) && (new_sel_end == sel_end)) return 0; /* no action required */ sel_start = new_sel_start; sel_end = new_sel_end; /* Allocate a new buffer before freeing the old one ... */ bp = kmalloc((sel_end-sel_start)/2+1, GFP_ATOMIC); if (!bp) { speakup_clear_selection(); return -ENOMEM; } kfree(sel_buffer); sel_buffer = bp; obp = bp; for (i = sel_start; i <= sel_end; i += 2) { *bp = sel_pos(i); if (!ishardspace(*bp++)) obp = bp; if (!((i + 2) % vc->vc_size_row)) { /* strip trailing blanks from line and add newline, * unless non-space at end of line. */ if (obp != bp) { bp = obp; *bp++ = '\r'; } obp = bp; } } sel_buffer_lth = bp - sel_buffer; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
William Hubbs40495.73%125.00%
Samuel Thibault122.84%125.00%
Chris Yungmann51.18%125.00%
Aleksei Fedotov10.24%125.00%
Total422100.00%4100.00%

struct speakup_paste_work { struct work_struct work; struct tty_struct *tty; };
static void __speakup_paste_selection(struct work_struct *work) { struct speakup_paste_work *spw = container_of(work, struct speakup_paste_work, work); struct tty_struct *tty = xchg(&spw->tty, NULL); struct vc_data *vc = (struct vc_data *)tty->driver_data; int pasted = 0, count; struct tty_ldisc *ld; DECLARE_WAITQUEUE(wait, current); ld = tty_ldisc_ref(tty); if (!ld) goto tty_unref; tty_buffer_lock_exclusive(&vc->port); add_wait_queue(&vc->paste_wait, &wait); while (sel_buffer && sel_buffer_lth > pasted) { set_current_state(TASK_INTERRUPTIBLE); if (tty_throttled(tty)) { schedule(); continue; } count = sel_buffer_lth - pasted; count = tty_ldisc_receive_buf(ld, sel_buffer + pasted, NULL, count); pasted += count; } remove_wait_queue(&vc->paste_wait, &wait); __set_current_state(TASK_RUNNING); tty_buffer_unlock_exclusive(&vc->port); tty_ldisc_deref(ld); tty_unref: tty_kref_put(tty); }

Contributors

PersonTokensPropCommitsCommitProp
William Hubbs10352.28%114.29%
Ben Hutchings7839.59%228.57%
Peter Hurley126.09%228.57%
Davidlohr Bueso A31.52%114.29%
Chris Yungmann10.51%114.29%
Total197100.00%7100.00%

static struct speakup_paste_work speakup_paste_work = { .work = __WORK_INITIALIZER(speakup_paste_work.work, __speakup_paste_selection) };
int speakup_paste_selection(struct tty_struct *tty) { if (cmpxchg(&speakup_paste_work.tty, NULL, tty) != NULL) return -EBUSY; tty_kref_get(tty); schedule_work_on(WORK_CPU_UNBOUND, &speakup_paste_work.work); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Ben Hutchings4491.67%150.00%
William Hubbs48.33%150.00%
Total48100.00%2100.00%


void speakup_cancel_paste(void) { cancel_work_sync(&speakup_paste_work.work); tty_kref_put(speakup_paste_work.tty); }

Contributors

PersonTokensPropCommitsCommitProp
Ben Hutchings22100.00%1100.00%
Total22100.00%1100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
William Hubbs67674.94%18.33%
Ben Hutchings18220.18%216.67%
Samuel Thibault161.77%18.33%
Peter Hurley121.33%216.67%
Chris Yungmann60.67%18.33%
George Spelvin40.44%18.33%
Davidlohr Bueso A30.33%18.33%
Boqun Feng10.11%18.33%
Aleksei Fedotov10.11%18.33%
Valentin Ilie10.11%18.33%
Total902100.00%12100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.