On Sat, Dec 20, 2014 at 03:19:44PM +0100, Nicolas Iooss wrote: > 2014-12-20 14:10 GMT+01:00 Jason Zaman: > > open_init_pty uses select() to handle all the file descriptors. There is > > a very high CPU usage due to select() always returning immediately with > > the fd is available for write. This uses a ring buffer and only calls > > select on the read/write fds that have data that needs to be > > read/written which eliminates the high CPU usage. > > > > This also correctly returns the exit code from the child process. > > > > This was originally from debian where they have been carrying it as a > > patch for a long time. Then we got a bug report in gentoo which this > > also happens to fix. The original debian patch had the ring buffer > > written in C++ so I modified the class into a struct and some static > > methods so it is C-only at the request of Steve Lawrence. > > > > Debian bug: https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=474956 > > Gentoo bug: https://bugs.gentoo.org/show_bug.cgi?id=532616 > > > > Signed-off-by: Jason Zaman <jason@xxxxxxxxxxxxx> > > Tested-by: Laurent Bigonville <bigon@xxxxxxxx> > > --- > > policycoreutils/run_init/open_init_pty.c | 511 ++++++++++++++++--------------- > > 1 file changed, 265 insertions(+), 246 deletions(-) > > > > diff --git a/policycoreutils/run_init/open_init_pty.c b/policycoreutils/run_init/open_init_pty.c > > index 4f04e72..fb428a3 100644 > > [SNIP] > > +static void setfd_nonblock(int fd) > > +{ > > + int fsflags = fcntl(fd, F_GETFL); > > + > > + if (fsflags < 0) { > > + fprintf(stderr, "fcntl(%d, F_GETFL): %s\n", fd, strerror(errno)); > > + exit(EX_IOERR); > > + } > > + > > + if (fcntl(STDIN_FILENO, F_SETFL, fsflags | O_NONBLOCK) < 0) { > > + fprintf(stderr, "fcntl(%d, F_SETFL, ... | O_NONBLOCK): %s\n", fd, strerror(errno)); > > + exit(EX_IOERR); > > + } > > +} > > Why does the second fcntl use STDIN_FILENO instead of fd? This indeed looks wrong, good catch. I did not touch this part so it must have worked in debian for quite a while. > > + > > +static void sigchld_handler(int asig __attribute__ ((unused))) > > +{ > > +} > > + > > int main(int argc, char *argv[]) > > { > > pid_t child_pid; > > [SNIP] > > > > - if (isatty(fileno(stdin))) { > > + if (isatty(STDIN_FILENO)) { > > /* get terminal parameters associated with stdout */ > > - if (tcgetattr(fileno(stdout), &tty_attr) < 0) { > > - perror("tcgetattr:"); > > + if (tcgetattr(STDOUT_FILENO, &tty_attr) < 0) { > > + perror("tcgetattr(stdout,...)"); > > exit(EX_OSERR); > > } > > > > - /* end of if(tcsetattr(&tty_attr)) */ > > /* get window size */ > > - if (ioctl(fileno(stdout), TIOCGWINSZ, &window_size) < 0) { > > - perror("ioctl stdout:"); > > + if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &window_size) < 0) { > > + perror("ioctl(stdout,...)"); > > exit(1); > > } > > Here, it seems quite weird to check whether stdin is a tty and then use > tcgetattr on stdout. As this behavior was in the initial code I > understand that this patch doesn't change it. For curiosity, can this > program happen to be called with stdin from a file but not stdout, or > the other way round? Indeed I did not write this but from what I understand, forkpty takes the tty attrs and size to make the window correctly sized so this part just checks if it is called interactively (if it is not then the window size does not make sense). the other case in the if statement jsut calls NULL. I think it is just so the child process knows but is not vital. > > + int select_rc = select(pty_master + 1, &readfds, &writefds, NULL, NULL); > > + if (select_rc < 0) { > > + perror("select()"); > > exit(EX_IOERR); > > } > > I get this error, with my custom CFLAGS: > > open_init_pty.c: In function 'main': > open_init_pty.c:330:3: error: ISO C90 forbids mixed declarations and > code [-Werror=pedantic] > int select_rc = select(pty_master + 1, &readfds, &writefds, NULL, NULL); > ^ > > I don't know what C specification follows SELinux userspace code but > there is a potential minor coding convention issue here. Feel free to > ignore this comment if this coding style (of not moving declarations to > the beginning of functions) is ok. The previous version had a lot of these type declarations and this version has more than just that single one. I wouldnt mind moving all the declarations to the beginning if it matches the style better tho. > Otherwise the patch looks good to me. I have not tested it as I'm using > systemd on my systems running SELinux and I believe it does not use > open_init_pty. you can test this by building and copying it to /usr/sbin/open_init_pty. run_init has that path hard coded so you would have to copy it there. You can then test it with "run_init /bin/echo hello" or "run_init id -Z" Thanks for the comments. I will wait a bit for any others then I'll send an updated version. -- Jason _______________________________________________ Selinux mailing list Selinux@xxxxxxxxxxxxx To unsubscribe, send email to Selinux-leave@xxxxxxxxxxxxx. To get help, send an email containing "help" to Selinux-request@xxxxxxxxxxxxx.