swaybar: Replace fgets with read and own buffer
This commit is contained in:
parent
b66c51ea2c
commit
21541e9e64
221
swaybar/main.c
221
swaybar/main.c
|
@ -1,3 +1,4 @@
|
||||||
|
#include <fcntl.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
@ -60,9 +61,9 @@ list_t *status_line = NULL;
|
||||||
list_t *workspaces = NULL;
|
list_t *workspaces = NULL;
|
||||||
int socketfd;
|
int socketfd;
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
int pipefd[2];
|
int status_read_fd;
|
||||||
FILE *command;
|
|
||||||
char line[1024];
|
char line[1024];
|
||||||
|
char line_rest[1024];
|
||||||
char *output, *status_command;
|
char *output, *status_command;
|
||||||
struct registry *registry;
|
struct registry *registry;
|
||||||
struct window *window;
|
struct window *window;
|
||||||
|
@ -110,10 +111,11 @@ struct {
|
||||||
int bufsize;
|
int bufsize;
|
||||||
char *buffer;
|
char *buffer;
|
||||||
char *line_start;
|
char *line_start;
|
||||||
|
char *parserpos;
|
||||||
bool escape;
|
bool escape;
|
||||||
int depth;
|
int depth;
|
||||||
int state[I3JSON_MAXDEPTH+1];
|
int state[I3JSON_MAXDEPTH+1];
|
||||||
} i3json_state = { 0, NULL, NULL, false, 0, { I3JSON_UNKNOWN } };
|
} i3json_state = { 0, NULL, NULL, NULL, false, 0, { I3JSON_UNKNOWN } };
|
||||||
|
|
||||||
|
|
||||||
void swaybar_teardown() {
|
void swaybar_teardown() {
|
||||||
|
@ -122,8 +124,8 @@ void swaybar_teardown() {
|
||||||
registry_teardown(registry);
|
registry_teardown(registry);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (command) {
|
if (status_read_fd) {
|
||||||
fclose(command);
|
close(status_read_fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pid) {
|
if (pid) {
|
||||||
|
@ -137,8 +139,8 @@ void swaybar_teardown() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pipefd[0]) {
|
if (status_read_fd) {
|
||||||
close(pipefd[0]);
|
close(status_read_fd);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -607,34 +609,124 @@ void parse_json(const char *text) {
|
||||||
json_object_put(results);
|
json_object_put(results);
|
||||||
}
|
}
|
||||||
|
|
||||||
int i3json_handle(FILE *file) {
|
// Read line from file descriptor, only show the line tail if it is too long.
|
||||||
char *c;
|
// In non-blocking mode treat "no more data" as a linebreak.
|
||||||
int handled = 0;
|
// If data after a line break has been read, return it in rest.
|
||||||
// handle partially buffered data
|
// If rest is non-empty, then use that as the start of the next line.
|
||||||
// make sure 1023+1 bytes at the end are free
|
int read_line_tail(int fd, char *buf, int nbyte, char *rest) {
|
||||||
if (i3json_state.line_start) {
|
if (fd < 0 || !buf || !nbyte) {
|
||||||
int len = strlen(i3json_state.line_start);
|
return -1;
|
||||||
memmove(i3json_state.buffer, i3json_state.line_start, len+1);
|
}
|
||||||
if (i3json_state.bufsize < len+1024) {
|
int l;
|
||||||
i3json_state.bufsize += 1024;
|
char *buffer = malloc(nbyte*2+1);
|
||||||
i3json_state.buffer = realloc(i3json_state.buffer, i3json_state.bufsize);
|
char *readpos = buffer;
|
||||||
|
char *lf;
|
||||||
|
// prepend old data to new line if necessary
|
||||||
|
if (rest) {
|
||||||
|
l = strlen(rest);
|
||||||
|
if (l > nbyte) {
|
||||||
|
strcpy(buffer, rest + l - nbyte);
|
||||||
|
readpos += nbyte;
|
||||||
|
} else if (l) {
|
||||||
|
strcpy(buffer, rest);
|
||||||
|
readpos += l;
|
||||||
}
|
}
|
||||||
c = i3json_state.buffer+len;
|
}
|
||||||
i3json_state.line_start = i3json_state.buffer;
|
// read until a linefeed is found or no more data is available
|
||||||
} else if (!i3json_state.buffer) {
|
while ((l = read(fd, readpos, nbyte)) > 0) {
|
||||||
i3json_state.buffer = malloc(1024);
|
readpos[l] = '\0';
|
||||||
i3json_state.bufsize = 1024;
|
lf = strchr(readpos, '\n');
|
||||||
c = i3json_state.buffer;
|
if (lf) {
|
||||||
|
// linefeed found, replace with \0
|
||||||
|
*lf = '\0';
|
||||||
|
// give data from the end of the line, try to fill the buffer
|
||||||
|
if (lf-buffer > nbyte) {
|
||||||
|
strcpy(buf, lf - nbyte + 1);
|
||||||
|
} else {
|
||||||
|
strcpy(buf, buffer);
|
||||||
|
}
|
||||||
|
// we may have read data from the next line, save it to rest
|
||||||
|
if (rest) {
|
||||||
|
rest[0] = '\0';
|
||||||
|
strcpy(rest, lf + 1);
|
||||||
|
}
|
||||||
|
free(buffer);
|
||||||
|
return strlen(buf);
|
||||||
|
} else {
|
||||||
|
// no linefeed found, slide data back.
|
||||||
|
int overflow = readpos - buffer + l - nbyte;
|
||||||
|
if (overflow > 0) {
|
||||||
|
memmove(buffer, buffer + overflow , nbyte + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (l < 0) {
|
||||||
|
free(buffer);
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
readpos[l]='\0';
|
||||||
|
if (rest) {
|
||||||
|
rest[0] = '\0';
|
||||||
|
}
|
||||||
|
if (nbyte < readpos - buffer + l - 1) {
|
||||||
|
memcpy(buf, readpos - nbyte + l + 1, nbyte);
|
||||||
} else {
|
} else {
|
||||||
c = i3json_state.buffer;
|
strncpy(buf, buffer, nbyte);
|
||||||
|
}
|
||||||
|
buf[nbyte-1] = '\0';
|
||||||
|
free(buffer);
|
||||||
|
return strlen(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
// make sure that enough buffer space is available starting from parserpos
|
||||||
|
void i3json_ensure_free(int min_free) {
|
||||||
|
int _step = 10240;
|
||||||
|
int r = min_free % _step;
|
||||||
|
if (r) {
|
||||||
|
min_free += _step - r;
|
||||||
|
}
|
||||||
|
if (!i3json_state.buffer) {
|
||||||
|
i3json_state.buffer = malloc(min_free);
|
||||||
|
i3json_state.bufsize = min_free;
|
||||||
|
i3json_state.parserpos = i3json_state.buffer;
|
||||||
|
} else {
|
||||||
|
int len = 0;
|
||||||
|
int pos = 0;
|
||||||
|
if (i3json_state.line_start) {
|
||||||
|
len = strlen(i3json_state.line_start);
|
||||||
|
pos = i3json_state.parserpos - i3json_state.line_start;
|
||||||
|
if (i3json_state.line_start != i3json_state.buffer) {
|
||||||
|
memmove(i3json_state.buffer, i3json_state.line_start, len+1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
len = strlen(i3json_state.buffer);
|
||||||
|
}
|
||||||
|
if (i3json_state.bufsize < len+min_free) {
|
||||||
|
i3json_state.bufsize += min_free;
|
||||||
|
if (i3json_state.bufsize > 1024000) {
|
||||||
|
sway_abort("Status line json too long or malformed.");
|
||||||
|
}
|
||||||
|
i3json_state.buffer = realloc(i3json_state.buffer, i3json_state.bufsize);
|
||||||
|
if (!i3json_state.buffer) {
|
||||||
|
sway_abort("Could not allocate json buffer");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (i3json_state.line_start) {
|
||||||
|
i3json_state.line_start = i3json_state.buffer;
|
||||||
|
i3json_state.parserpos = i3json_state.buffer + pos;
|
||||||
|
} else {
|
||||||
|
i3json_state.parserpos = i3json_state.buffer;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (!i3json_state.buffer) {
|
if (!i3json_state.buffer) {
|
||||||
sway_abort("Could not allocate buffer.");
|
sway_abort("Could not allocate buffer.");
|
||||||
}
|
}
|
||||||
// get fresh data at the end of the buffer
|
}
|
||||||
if (!fgets(c, 1023, file)) return -1;
|
|
||||||
c[1023] = '\0';
|
|
||||||
|
|
||||||
|
// continue parsing from last parserpos
|
||||||
|
int i3json_parse() {
|
||||||
|
char *c = i3json_state.parserpos;
|
||||||
|
int handled = 0;
|
||||||
while (*c) {
|
while (*c) {
|
||||||
if (i3json_state.state[i3json_state.depth] == I3JSON_STRING) {
|
if (i3json_state.state[i3json_state.depth] == I3JSON_STRING) {
|
||||||
if (!i3json_state.escape && *c == '"') {
|
if (!i3json_state.escape && *c == '"') {
|
||||||
|
@ -679,9 +771,30 @@ int i3json_handle(FILE *file) {
|
||||||
}
|
}
|
||||||
++c;
|
++c;
|
||||||
}
|
}
|
||||||
|
i3json_state.parserpos = c;
|
||||||
return handled;
|
return handled;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// append data and parse it.
|
||||||
|
int i3json_handle_data(char *data) {
|
||||||
|
int len = strlen(data);
|
||||||
|
i3json_ensure_free(len);
|
||||||
|
strcpy(i3json_state.parserpos, data);
|
||||||
|
return i3json_parse();
|
||||||
|
}
|
||||||
|
|
||||||
|
// read data from fd and parse it.
|
||||||
|
int i3json_handle_fd(int fd) {
|
||||||
|
i3json_ensure_free(10240);
|
||||||
|
// get fresh data at the end of the buffer
|
||||||
|
int readlen = read(fd, i3json_state.parserpos, 10239);
|
||||||
|
if (readlen < 0) {
|
||||||
|
return readlen;
|
||||||
|
}
|
||||||
|
i3json_state.parserpos[readlen] = '\0';
|
||||||
|
return i3json_parse();
|
||||||
|
}
|
||||||
|
|
||||||
void poll_for_update() {
|
void poll_for_update() {
|
||||||
fd_set readfds;
|
fd_set readfds;
|
||||||
int activity;
|
int activity;
|
||||||
|
@ -698,7 +811,7 @@ void poll_for_update() {
|
||||||
dirty = false;
|
dirty = false;
|
||||||
FD_ZERO(&readfds);
|
FD_ZERO(&readfds);
|
||||||
FD_SET(socketfd, &readfds);
|
FD_SET(socketfd, &readfds);
|
||||||
FD_SET(pipefd[0], &readfds);
|
FD_SET(status_read_fd, &readfds);
|
||||||
|
|
||||||
activity = select(FD_SETSIZE, &readfds, NULL, NULL, NULL);
|
activity = select(FD_SETSIZE, &readfds, NULL, NULL, NULL);
|
||||||
if (activity < 0) {
|
if (activity < 0) {
|
||||||
|
@ -714,41 +827,40 @@ void poll_for_update() {
|
||||||
dirty = true;
|
dirty = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (status_command && FD_ISSET(pipefd[0], &readfds)) {
|
if (status_command && FD_ISSET(status_read_fd, &readfds)) {
|
||||||
sway_log(L_DEBUG, "Got update from status command.");
|
sway_log(L_DEBUG, "Got update from status command.");
|
||||||
int linelen;
|
|
||||||
switch (protocol) {
|
switch (protocol) {
|
||||||
case I3BAR:
|
case I3BAR:
|
||||||
sway_log(L_DEBUG, "Got i3bar protocol.");
|
sway_log(L_DEBUG, "Got i3bar protocol.");
|
||||||
if (i3json_handle(command) > 0) {
|
if (i3json_handle_fd(status_read_fd) > 0) {
|
||||||
dirty = true;
|
dirty = true;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case TEXT:
|
case TEXT:
|
||||||
case UNDEF:
|
|
||||||
sway_log(L_DEBUG, "Got text protocol.");
|
sway_log(L_DEBUG, "Got text protocol.");
|
||||||
fgets(line, sizeof(line), command);
|
read_line_tail(status_read_fd, line, sizeof(line), line_rest);
|
||||||
linelen = strlen(line) - 1;
|
dirty = true;
|
||||||
if (line[linelen] == '\n') {
|
break;
|
||||||
line[linelen] = '\0';
|
case UNDEF:
|
||||||
|
sway_log(L_DEBUG, "Detecting protocol...");
|
||||||
|
if (read_line_tail(status_read_fd, line, sizeof(line), line_rest) < 0) {
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
dirty = true;
|
dirty = true;
|
||||||
if (protocol == UNDEF) {
|
protocol = TEXT;
|
||||||
protocol = TEXT;
|
if (line[0] == '{') {
|
||||||
if (line[0] == '{') {
|
// detect i3bar json protocol
|
||||||
// detect i3bar json protocol
|
json_object *proto = json_tokener_parse(line);
|
||||||
json_object *proto = json_tokener_parse(line);
|
json_object *version;
|
||||||
json_object *version;
|
if (proto) {
|
||||||
if (proto) {
|
if (json_object_object_get_ex(proto, "version", &version)
|
||||||
if (json_object_object_get_ex(proto, "version", &version)
|
&& json_object_get_int(version) == 1
|
||||||
&& json_object_get_int(version) == 1
|
) {
|
||||||
) {
|
sway_log(L_DEBUG, "Switched to i3bar protocol.");
|
||||||
sway_log(L_DEBUG, "Switched to i3bar protocol.");
|
protocol = I3BAR;
|
||||||
protocol = I3BAR;
|
i3json_handle_data(line_rest);
|
||||||
line[0] = '\0';
|
|
||||||
}
|
|
||||||
json_object_put(proto);
|
|
||||||
}
|
}
|
||||||
|
json_object_put(proto);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -831,6 +943,7 @@ int main(int argc, char **argv) {
|
||||||
bar_ipc_init(desired_output, bar_id);
|
bar_ipc_init(desired_output, bar_id);
|
||||||
|
|
||||||
if (status_command) {
|
if (status_command) {
|
||||||
|
int pipefd[2];
|
||||||
pipe(pipefd);
|
pipe(pipefd);
|
||||||
pid = fork();
|
pid = fork();
|
||||||
if (pid == 0) {
|
if (pid == 0) {
|
||||||
|
@ -848,8 +961,8 @@ int main(int argc, char **argv) {
|
||||||
}
|
}
|
||||||
|
|
||||||
close(pipefd[1]);
|
close(pipefd[1]);
|
||||||
command = fdopen(pipefd[0], "r");
|
status_read_fd = pipefd[0];
|
||||||
setbuf(command, NULL);
|
fcntl(status_read_fd, F_SETFL, O_NONBLOCK);
|
||||||
line[0] = '\0';
|
line[0] = '\0';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue