-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathtty-resizer.bpf.c
74 lines (63 loc) · 2.02 KB
/
tty-resizer.bpf.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
/*
* Copyright (C) 2023, Yasumasa Suenaga
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*/
#include "vmlinux.h"
#include <bpf/bpf_tracing.h>
#include <bpf/bpf_helpers.h>
#include <bpf/bpf_core_read.h>
#include "common.h"
char LICENSE[] SEC("license") = "GPL";
volatile ino_t tty_ino = -1;
volatile bool send_to_buffer = false;
const int ringbuf_sz = RINGBUF_SZ;
struct {
__uint(type, BPF_MAP_TYPE_RINGBUF);
__uint(max_entries, RINGBUF_SZ);
} received_chars SEC(".maps");
SEC("fexit/tty_read")
int BPF_PROG(tty_read, struct kiocb *iocb, struct iov_iter *to, int ret){
char ch;
void *ubuf;
if (BPF_CORE_READ(iocb, ki_filp, f_inode, i_ino) != tty_ino){
return 0;
}
ubuf = BPF_CORE_READ(to, ubuf);
bpf_probe_read_user(&ch, 1, ubuf);
if(ch == RESIZER_START_MARKER){
send_to_buffer = true;
ch = '\0';
bpf_probe_write_user(ubuf, &ch, 1);
}
else if(send_to_buffer){
u64 flags = BPF_RB_NO_WAKEUP;
if(ch == RESIZER_END_MARKER){
flags = BPF_RB_FORCE_WAKEUP;
send_to_buffer = false;
}
else if((ch != *RESIZER_SEPARATOR) && (('0' > ch) || ('9' < ch))){
/* handles invalid chars */
ch = RESIZER_CANCEL_MARKER;
flags = BPF_RB_FORCE_WAKEUP;
send_to_buffer = false;
}
bpf_ringbuf_output(&received_chars, &ch, 1, flags);
ch = '\0';
bpf_probe_write_user(ubuf, &ch, 1);
}
return 0;
}