How to send ^D / EOT character to stdin of a shell process?












-1















Suppose I have two shells open, one with PID 1234 and the other - 5678.



I can clear the screen and reset of the other shell via echo $'33c' > /proc/1234/fd/0 as if reset was typed on the prompt of the terminal itself. However, when I do echo $'4' > /proc/1234/fd/0 to send EOT character, this doesn't seem to work. How can I achieve the desired effect ?










share|improve this question


















  • 2





    Perhaps send SIGHUP, since you cannot send EOF.

    – Thomas Dickey
    Jan 10 at 0:48











  • @ThomasDickey EOF is not a character, EOT however is actually on the list of ASCII characters, hex 04 value. That's what I'm trying to send. Heemayls answer there isn't accurate, I think.

    – Sergiy Kolodyazhnyy
    Jan 10 at 0:56













  • Maybe because you're in canonical mode so such bytes would be consumed by tty driver before it get send.

    – 炸鱼薯条德里克
    Jan 10 at 0:56













  • @炸鱼 OK, how do I check if I'm canonical mode ?

    – Sergiy Kolodyazhnyy
    Jan 10 at 1:00











  • stty -a will dump the terminal settings for the current terminal. In the flags portion you will see icanon, the canonical flag. If this flag is preceded by an - then it is off, no dash then its on.

    – datUser
    Jan 10 at 1:27
















-1















Suppose I have two shells open, one with PID 1234 and the other - 5678.



I can clear the screen and reset of the other shell via echo $'33c' > /proc/1234/fd/0 as if reset was typed on the prompt of the terminal itself. However, when I do echo $'4' > /proc/1234/fd/0 to send EOT character, this doesn't seem to work. How can I achieve the desired effect ?










share|improve this question


















  • 2





    Perhaps send SIGHUP, since you cannot send EOF.

    – Thomas Dickey
    Jan 10 at 0:48











  • @ThomasDickey EOF is not a character, EOT however is actually on the list of ASCII characters, hex 04 value. That's what I'm trying to send. Heemayls answer there isn't accurate, I think.

    – Sergiy Kolodyazhnyy
    Jan 10 at 0:56













  • Maybe because you're in canonical mode so such bytes would be consumed by tty driver before it get send.

    – 炸鱼薯条德里克
    Jan 10 at 0:56













  • @炸鱼 OK, how do I check if I'm canonical mode ?

    – Sergiy Kolodyazhnyy
    Jan 10 at 1:00











  • stty -a will dump the terminal settings for the current terminal. In the flags portion you will see icanon, the canonical flag. If this flag is preceded by an - then it is off, no dash then its on.

    – datUser
    Jan 10 at 1:27














-1












-1








-1








Suppose I have two shells open, one with PID 1234 and the other - 5678.



I can clear the screen and reset of the other shell via echo $'33c' > /proc/1234/fd/0 as if reset was typed on the prompt of the terminal itself. However, when I do echo $'4' > /proc/1234/fd/0 to send EOT character, this doesn't seem to work. How can I achieve the desired effect ?










share|improve this question














Suppose I have two shells open, one with PID 1234 and the other - 5678.



I can clear the screen and reset of the other shell via echo $'33c' > /proc/1234/fd/0 as if reset was typed on the prompt of the terminal itself. However, when I do echo $'4' > /proc/1234/fd/0 to send EOT character, this doesn't seem to work. How can I achieve the desired effect ?







command-line escape-characters






share|improve this question













share|improve this question











share|improve this question




share|improve this question










asked Jan 9 at 23:58









Sergiy KolodyazhnyySergiy Kolodyazhnyy

8,56912254




8,56912254








  • 2





    Perhaps send SIGHUP, since you cannot send EOF.

    – Thomas Dickey
    Jan 10 at 0:48











  • @ThomasDickey EOF is not a character, EOT however is actually on the list of ASCII characters, hex 04 value. That's what I'm trying to send. Heemayls answer there isn't accurate, I think.

    – Sergiy Kolodyazhnyy
    Jan 10 at 0:56













  • Maybe because you're in canonical mode so such bytes would be consumed by tty driver before it get send.

    – 炸鱼薯条德里克
    Jan 10 at 0:56













  • @炸鱼 OK, how do I check if I'm canonical mode ?

    – Sergiy Kolodyazhnyy
    Jan 10 at 1:00











  • stty -a will dump the terminal settings for the current terminal. In the flags portion you will see icanon, the canonical flag. If this flag is preceded by an - then it is off, no dash then its on.

    – datUser
    Jan 10 at 1:27














  • 2





    Perhaps send SIGHUP, since you cannot send EOF.

    – Thomas Dickey
    Jan 10 at 0:48











  • @ThomasDickey EOF is not a character, EOT however is actually on the list of ASCII characters, hex 04 value. That's what I'm trying to send. Heemayls answer there isn't accurate, I think.

    – Sergiy Kolodyazhnyy
    Jan 10 at 0:56













  • Maybe because you're in canonical mode so such bytes would be consumed by tty driver before it get send.

    – 炸鱼薯条德里克
    Jan 10 at 0:56













  • @炸鱼 OK, how do I check if I'm canonical mode ?

    – Sergiy Kolodyazhnyy
    Jan 10 at 1:00











  • stty -a will dump the terminal settings for the current terminal. In the flags portion you will see icanon, the canonical flag. If this flag is preceded by an - then it is off, no dash then its on.

    – datUser
    Jan 10 at 1:27








2




2





Perhaps send SIGHUP, since you cannot send EOF.

– Thomas Dickey
Jan 10 at 0:48





Perhaps send SIGHUP, since you cannot send EOF.

– Thomas Dickey
Jan 10 at 0:48













@ThomasDickey EOF is not a character, EOT however is actually on the list of ASCII characters, hex 04 value. That's what I'm trying to send. Heemayls answer there isn't accurate, I think.

– Sergiy Kolodyazhnyy
Jan 10 at 0:56







@ThomasDickey EOF is not a character, EOT however is actually on the list of ASCII characters, hex 04 value. That's what I'm trying to send. Heemayls answer there isn't accurate, I think.

– Sergiy Kolodyazhnyy
Jan 10 at 0:56















Maybe because you're in canonical mode so such bytes would be consumed by tty driver before it get send.

– 炸鱼薯条德里克
Jan 10 at 0:56







Maybe because you're in canonical mode so such bytes would be consumed by tty driver before it get send.

– 炸鱼薯条德里克
Jan 10 at 0:56















@炸鱼 OK, how do I check if I'm canonical mode ?

– Sergiy Kolodyazhnyy
Jan 10 at 1:00





@炸鱼 OK, how do I check if I'm canonical mode ?

– Sergiy Kolodyazhnyy
Jan 10 at 1:00













stty -a will dump the terminal settings for the current terminal. In the flags portion you will see icanon, the canonical flag. If this flag is preceded by an - then it is off, no dash then its on.

– datUser
Jan 10 at 1:27





stty -a will dump the terminal settings for the current terminal. In the flags portion you will see icanon, the canonical flag. If this flag is preceded by an - then it is off, no dash then its on.

– datUser
Jan 10 at 1:27










1 Answer
1






active

oldest

votes


















1














First of all, a tty, even if it appears a single file object, is actually a pair of pipes/queues -- one for the output and one for the input, the other ends of each being connected either to the TX/RX of some hardware device, or made available to another program in the case of a pseudo-tty. When you're writing anything to the fd linked to a tty, you're writing into the output queue, even if that's the fd 0 (stdin) of your process, and will not be looped back into the input queue.



There's a special ioctl (TIOCSTI) which allows you to insert a byte into the input queue of a tty, as if it were received from the other end. TIOCSTI will only work as root when not used on the controlling tty of the process calling it.



$ cc -Wall tiocsti.c -o tiocsti


-- as root --



# printf '4' | ./tiocsti >/proc/5460/fd/0


If your process is running in a pseudo-tty, another way to "fake" input is to connect with a debugger to the process holding the master side of the pseudo-tty, and write(2) your stuff into it from there.



$ gdb -p pid_of_xterm
(gdb) p write(4, "x04", 1) # 4 is the fd of the master pty
(gdb) c


Both methods are just overkill. If you want to control a running process, you better attach with a debugger to that very process instead of faking it some input; if you want to instrumentalize an interactive program, you better run it in expect; both will work on any Unix system and don't require extra privileges for that.



Also notice that ^D / 04 will only work as EOF if a) the tty is in canonical mode b) the c_cc[VEOF] special character wasn't changed to something else c) the output queue is empty (otherwise c_cc[VEOF] will have to be sent twice). And of course, it will be the process which is just reading from tty which will get the EOF or those bytes, not necessarily the shell.



Note:



As to why anyone would ever want to change VEOF to something else than ^D here is a case: if you want to input a password from the user, you turn echo off, keep canonical mode on (so the user be able to backspace, etc), set VEOL2 to ^D (a ^D will end the password, just like a newline), and VEOF to ^C (a ^C will abort the password, and you will get that as 0 return from read). That way, you can implement getpass(3) without having to mess with any signals.



tiocsti.c



#include <sys/ioctl.h>
#include <unistd.h>
#include <err.h>

int main(void){
char c; ssize_t r;
for(;;){
if((r = read(0, &c, 1)) <= 0){
if(r < 0) err(1, "read");
return 0;
}
if(ioctl(1, TIOCSTI, &c)) err(1, "ioctl(TIOCSTI)");
}
}





share|improve this answer

























    Your Answer








    StackExchange.ready(function() {
    var channelOptions = {
    tags: "".split(" "),
    id: "106"
    };
    initTagRenderer("".split(" "), "".split(" "), channelOptions);

    StackExchange.using("externalEditor", function() {
    // Have to fire editor after snippets, if snippets enabled
    if (StackExchange.settings.snippets.snippetsEnabled) {
    StackExchange.using("snippets", function() {
    createEditor();
    });
    }
    else {
    createEditor();
    }
    });

    function createEditor() {
    StackExchange.prepareEditor({
    heartbeatType: 'answer',
    autoActivateHeartbeat: false,
    convertImagesToLinks: false,
    noModals: true,
    showLowRepImageUploadWarning: true,
    reputationToPostImages: null,
    bindNavPrevention: true,
    postfix: "",
    imageUploader: {
    brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
    contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
    allowUrls: true
    },
    onDemand: true,
    discardSelector: ".discard-answer"
    ,immediatelyShowMarkdownHelp:true
    });


    }
    });














    draft saved

    draft discarded


















    StackExchange.ready(
    function () {
    StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2funix.stackexchange.com%2fquestions%2f493578%2fhow-to-send-d-eot-character-to-stdin-of-a-shell-process%23new-answer', 'question_page');
    }
    );

    Post as a guest















    Required, but never shown

























    1 Answer
    1






    active

    oldest

    votes








    1 Answer
    1






    active

    oldest

    votes









    active

    oldest

    votes






    active

    oldest

    votes









    1














    First of all, a tty, even if it appears a single file object, is actually a pair of pipes/queues -- one for the output and one for the input, the other ends of each being connected either to the TX/RX of some hardware device, or made available to another program in the case of a pseudo-tty. When you're writing anything to the fd linked to a tty, you're writing into the output queue, even if that's the fd 0 (stdin) of your process, and will not be looped back into the input queue.



    There's a special ioctl (TIOCSTI) which allows you to insert a byte into the input queue of a tty, as if it were received from the other end. TIOCSTI will only work as root when not used on the controlling tty of the process calling it.



    $ cc -Wall tiocsti.c -o tiocsti


    -- as root --



    # printf '4' | ./tiocsti >/proc/5460/fd/0


    If your process is running in a pseudo-tty, another way to "fake" input is to connect with a debugger to the process holding the master side of the pseudo-tty, and write(2) your stuff into it from there.



    $ gdb -p pid_of_xterm
    (gdb) p write(4, "x04", 1) # 4 is the fd of the master pty
    (gdb) c


    Both methods are just overkill. If you want to control a running process, you better attach with a debugger to that very process instead of faking it some input; if you want to instrumentalize an interactive program, you better run it in expect; both will work on any Unix system and don't require extra privileges for that.



    Also notice that ^D / 04 will only work as EOF if a) the tty is in canonical mode b) the c_cc[VEOF] special character wasn't changed to something else c) the output queue is empty (otherwise c_cc[VEOF] will have to be sent twice). And of course, it will be the process which is just reading from tty which will get the EOF or those bytes, not necessarily the shell.



    Note:



    As to why anyone would ever want to change VEOF to something else than ^D here is a case: if you want to input a password from the user, you turn echo off, keep canonical mode on (so the user be able to backspace, etc), set VEOL2 to ^D (a ^D will end the password, just like a newline), and VEOF to ^C (a ^C will abort the password, and you will get that as 0 return from read). That way, you can implement getpass(3) without having to mess with any signals.



    tiocsti.c



    #include <sys/ioctl.h>
    #include <unistd.h>
    #include <err.h>

    int main(void){
    char c; ssize_t r;
    for(;;){
    if((r = read(0, &c, 1)) <= 0){
    if(r < 0) err(1, "read");
    return 0;
    }
    if(ioctl(1, TIOCSTI, &c)) err(1, "ioctl(TIOCSTI)");
    }
    }





    share|improve this answer






























      1














      First of all, a tty, even if it appears a single file object, is actually a pair of pipes/queues -- one for the output and one for the input, the other ends of each being connected either to the TX/RX of some hardware device, or made available to another program in the case of a pseudo-tty. When you're writing anything to the fd linked to a tty, you're writing into the output queue, even if that's the fd 0 (stdin) of your process, and will not be looped back into the input queue.



      There's a special ioctl (TIOCSTI) which allows you to insert a byte into the input queue of a tty, as if it were received from the other end. TIOCSTI will only work as root when not used on the controlling tty of the process calling it.



      $ cc -Wall tiocsti.c -o tiocsti


      -- as root --



      # printf '4' | ./tiocsti >/proc/5460/fd/0


      If your process is running in a pseudo-tty, another way to "fake" input is to connect with a debugger to the process holding the master side of the pseudo-tty, and write(2) your stuff into it from there.



      $ gdb -p pid_of_xterm
      (gdb) p write(4, "x04", 1) # 4 is the fd of the master pty
      (gdb) c


      Both methods are just overkill. If you want to control a running process, you better attach with a debugger to that very process instead of faking it some input; if you want to instrumentalize an interactive program, you better run it in expect; both will work on any Unix system and don't require extra privileges for that.



      Also notice that ^D / 04 will only work as EOF if a) the tty is in canonical mode b) the c_cc[VEOF] special character wasn't changed to something else c) the output queue is empty (otherwise c_cc[VEOF] will have to be sent twice). And of course, it will be the process which is just reading from tty which will get the EOF or those bytes, not necessarily the shell.



      Note:



      As to why anyone would ever want to change VEOF to something else than ^D here is a case: if you want to input a password from the user, you turn echo off, keep canonical mode on (so the user be able to backspace, etc), set VEOL2 to ^D (a ^D will end the password, just like a newline), and VEOF to ^C (a ^C will abort the password, and you will get that as 0 return from read). That way, you can implement getpass(3) without having to mess with any signals.



      tiocsti.c



      #include <sys/ioctl.h>
      #include <unistd.h>
      #include <err.h>

      int main(void){
      char c; ssize_t r;
      for(;;){
      if((r = read(0, &c, 1)) <= 0){
      if(r < 0) err(1, "read");
      return 0;
      }
      if(ioctl(1, TIOCSTI, &c)) err(1, "ioctl(TIOCSTI)");
      }
      }





      share|improve this answer




























        1












        1








        1







        First of all, a tty, even if it appears a single file object, is actually a pair of pipes/queues -- one for the output and one for the input, the other ends of each being connected either to the TX/RX of some hardware device, or made available to another program in the case of a pseudo-tty. When you're writing anything to the fd linked to a tty, you're writing into the output queue, even if that's the fd 0 (stdin) of your process, and will not be looped back into the input queue.



        There's a special ioctl (TIOCSTI) which allows you to insert a byte into the input queue of a tty, as if it were received from the other end. TIOCSTI will only work as root when not used on the controlling tty of the process calling it.



        $ cc -Wall tiocsti.c -o tiocsti


        -- as root --



        # printf '4' | ./tiocsti >/proc/5460/fd/0


        If your process is running in a pseudo-tty, another way to "fake" input is to connect with a debugger to the process holding the master side of the pseudo-tty, and write(2) your stuff into it from there.



        $ gdb -p pid_of_xterm
        (gdb) p write(4, "x04", 1) # 4 is the fd of the master pty
        (gdb) c


        Both methods are just overkill. If you want to control a running process, you better attach with a debugger to that very process instead of faking it some input; if you want to instrumentalize an interactive program, you better run it in expect; both will work on any Unix system and don't require extra privileges for that.



        Also notice that ^D / 04 will only work as EOF if a) the tty is in canonical mode b) the c_cc[VEOF] special character wasn't changed to something else c) the output queue is empty (otherwise c_cc[VEOF] will have to be sent twice). And of course, it will be the process which is just reading from tty which will get the EOF or those bytes, not necessarily the shell.



        Note:



        As to why anyone would ever want to change VEOF to something else than ^D here is a case: if you want to input a password from the user, you turn echo off, keep canonical mode on (so the user be able to backspace, etc), set VEOL2 to ^D (a ^D will end the password, just like a newline), and VEOF to ^C (a ^C will abort the password, and you will get that as 0 return from read). That way, you can implement getpass(3) without having to mess with any signals.



        tiocsti.c



        #include <sys/ioctl.h>
        #include <unistd.h>
        #include <err.h>

        int main(void){
        char c; ssize_t r;
        for(;;){
        if((r = read(0, &c, 1)) <= 0){
        if(r < 0) err(1, "read");
        return 0;
        }
        if(ioctl(1, TIOCSTI, &c)) err(1, "ioctl(TIOCSTI)");
        }
        }





        share|improve this answer















        First of all, a tty, even if it appears a single file object, is actually a pair of pipes/queues -- one for the output and one for the input, the other ends of each being connected either to the TX/RX of some hardware device, or made available to another program in the case of a pseudo-tty. When you're writing anything to the fd linked to a tty, you're writing into the output queue, even if that's the fd 0 (stdin) of your process, and will not be looped back into the input queue.



        There's a special ioctl (TIOCSTI) which allows you to insert a byte into the input queue of a tty, as if it were received from the other end. TIOCSTI will only work as root when not used on the controlling tty of the process calling it.



        $ cc -Wall tiocsti.c -o tiocsti


        -- as root --



        # printf '4' | ./tiocsti >/proc/5460/fd/0


        If your process is running in a pseudo-tty, another way to "fake" input is to connect with a debugger to the process holding the master side of the pseudo-tty, and write(2) your stuff into it from there.



        $ gdb -p pid_of_xterm
        (gdb) p write(4, "x04", 1) # 4 is the fd of the master pty
        (gdb) c


        Both methods are just overkill. If you want to control a running process, you better attach with a debugger to that very process instead of faking it some input; if you want to instrumentalize an interactive program, you better run it in expect; both will work on any Unix system and don't require extra privileges for that.



        Also notice that ^D / 04 will only work as EOF if a) the tty is in canonical mode b) the c_cc[VEOF] special character wasn't changed to something else c) the output queue is empty (otherwise c_cc[VEOF] will have to be sent twice). And of course, it will be the process which is just reading from tty which will get the EOF or those bytes, not necessarily the shell.



        Note:



        As to why anyone would ever want to change VEOF to something else than ^D here is a case: if you want to input a password from the user, you turn echo off, keep canonical mode on (so the user be able to backspace, etc), set VEOL2 to ^D (a ^D will end the password, just like a newline), and VEOF to ^C (a ^C will abort the password, and you will get that as 0 return from read). That way, you can implement getpass(3) without having to mess with any signals.



        tiocsti.c



        #include <sys/ioctl.h>
        #include <unistd.h>
        #include <err.h>

        int main(void){
        char c; ssize_t r;
        for(;;){
        if((r = read(0, &c, 1)) <= 0){
        if(r < 0) err(1, "read");
        return 0;
        }
        if(ioctl(1, TIOCSTI, &c)) err(1, "ioctl(TIOCSTI)");
        }
        }






        share|improve this answer














        share|improve this answer



        share|improve this answer








        edited Jan 10 at 15:25

























        answered Jan 10 at 11:28









        mosvymosvy

        6,2811426




        6,2811426






























            draft saved

            draft discarded




















































            Thanks for contributing an answer to Unix & Linux Stack Exchange!


            • Please be sure to answer the question. Provide details and share your research!

            But avoid



            • Asking for help, clarification, or responding to other answers.

            • Making statements based on opinion; back them up with references or personal experience.


            To learn more, see our tips on writing great answers.




            draft saved


            draft discarded














            StackExchange.ready(
            function () {
            StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2funix.stackexchange.com%2fquestions%2f493578%2fhow-to-send-d-eot-character-to-stdin-of-a-shell-process%23new-answer', 'question_page');
            }
            );

            Post as a guest















            Required, but never shown





















































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown

































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown







            Popular posts from this blog

            Morgemoulin

            Scott Moir

            Souastre