1*16350Skarels /* $Header: /a/guest/moore/talk/RCS/invite.c,v 1.7 83/07/06 00:17:32 moore Exp $ */ 2*16350Skarels 3*16350Skarels #include "talk_ctl.h" 4*16350Skarels #include <sys/time.h> 5*16350Skarels #include <signal.h> 6*16350Skarels #include <setjmp.h> 7*16350Skarels 8*16350Skarels /* 9*16350Skarels * there wasn't an invitation waiting, so send a request containing 10*16350Skarels * our sockt address to the remote talk daemon so it can invite 11*16350Skarels * him 12*16350Skarels */ 13*16350Skarels 14*16350Skarels int local_id, remote_id; /* the msg.id's for the invitations 15*16350Skarels on the local and remote machines. 16*16350Skarels These are used to delete the 17*16350Skarels invitations. */ 18*16350Skarels void re_invite(); 19*16350Skarels jmp_buf invitebuf; 20*16350Skarels 21*16350Skarels invite_remote() 22*16350Skarels { 23*16350Skarels int nfd, read_mask, template, new_sockt; 24*16350Skarels struct itimerval itimer; 25*16350Skarels CTL_RESPONSE response; 26*16350Skarels 27*16350Skarels itimer.it_value.tv_sec = RING_WAIT; 28*16350Skarels itimer.it_value.tv_usec = 0; 29*16350Skarels itimer.it_interval = itimer.it_value; 30*16350Skarels 31*16350Skarels if (listen(sockt, 5) != 0) { 32*16350Skarels p_error("Error on attempt to listen for caller"); 33*16350Skarels } 34*16350Skarels 35*16350Skarels msg.addr = my_addr; 36*16350Skarels msg.id_num = -1; /* an impossible id_num */ 37*16350Skarels 38*16350Skarels invitation_waiting = 1; 39*16350Skarels 40*16350Skarels announce_invite(); 41*16350Skarels 42*16350Skarels /* 43*16350Skarels * shut off the automatic messages for a while, 44*16350Skarels * so we can use the interupt timer to resend the invitation 45*16350Skarels */ 46*16350Skarels 47*16350Skarels end_msgs(); 48*16350Skarels setitimer(ITIMER_REAL, &itimer, (struct itimerval *)0); 49*16350Skarels message("Waiting for your party to respond"); 50*16350Skarels signal(SIGALRM, re_invite); 51*16350Skarels (void) setjmp(invitebuf); 52*16350Skarels 53*16350Skarels while ((new_sockt = accept(sockt, 0, 0)) < 0) { 54*16350Skarels if (errno != EINTR) { 55*16350Skarels p_error("Unable to connect with your party"); 56*16350Skarels } else { 57*16350Skarels /* we just returned from a interupt, keep trying */ 58*16350Skarels continue; 59*16350Skarels } 60*16350Skarels } 61*16350Skarels 62*16350Skarels close(sockt); 63*16350Skarels sockt = new_sockt; 64*16350Skarels 65*16350Skarels /* have the daemons delete the invitations now that we 66*16350Skarels have connected. 67*16350Skarels */ 68*16350Skarels 69*16350Skarels current_state = "Waiting for your party to respond"; 70*16350Skarels start_msgs(); 71*16350Skarels 72*16350Skarels msg.id_num = local_id; 73*16350Skarels ctl_transact(my_machine_addr, msg, DELETE, &response); 74*16350Skarels msg.id_num = remote_id; 75*16350Skarels ctl_transact(his_machine_addr, msg, DELETE, &response); 76*16350Skarels invitation_waiting = 0; 77*16350Skarels } 78*16350Skarels 79*16350Skarels /* routine called on interupt to re-invite the callee */ 80*16350Skarels 81*16350Skarels void re_invite() 82*16350Skarels { 83*16350Skarels message("Ringing your party again"); 84*16350Skarels current_line++; 85*16350Skarels /* force a re-announce */ 86*16350Skarels msg.id_num = remote_id + 1; 87*16350Skarels announce_invite(); 88*16350Skarels longjmp(invitebuf, 1); 89*16350Skarels } 90*16350Skarels 91*16350Skarels /* transmit the invitation and process the response */ 92*16350Skarels 93*16350Skarels announce_invite() 94*16350Skarels { 95*16350Skarels CTL_RESPONSE response; 96*16350Skarels 97*16350Skarels current_state = "Trying to connect to your party's talk daemon"; 98*16350Skarels 99*16350Skarels ctl_transact(his_machine_addr, msg, ANNOUNCE, &response); 100*16350Skarels remote_id = response.id_num; 101*16350Skarels 102*16350Skarels if (response.answer != SUCCESS) { 103*16350Skarels 104*16350Skarels switch (response.answer) { 105*16350Skarels 106*16350Skarels case NOT_HERE : 107*16350Skarels message("Your party is not logged on"); 108*16350Skarels break; 109*16350Skarels 110*16350Skarels case MACHINE_UNKNOWN : 111*16350Skarels message("Target machine does not recognize us"); 112*16350Skarels break; 113*16350Skarels 114*16350Skarels case UNKNOWN_REQUEST : 115*16350Skarels message("Target machine can not handle remote talk"); 116*16350Skarels break; 117*16350Skarels 118*16350Skarels case FAILED : 119*16350Skarels message("Target machine is too confused to talk to us"); 120*16350Skarels break; 121*16350Skarels 122*16350Skarels case PERMISSION_DENIED : 123*16350Skarels message("Your party is refusing messages"); 124*16350Skarels break; 125*16350Skarels } 126*16350Skarels 127*16350Skarels quit(); 128*16350Skarels } 129*16350Skarels 130*16350Skarels /* leave the actual invitation on my talk daemon */ 131*16350Skarels 132*16350Skarels ctl_transact(my_machine_addr, msg, LEAVE_INVITE, &response); 133*16350Skarels local_id = response.id_num; 134*16350Skarels } 135*16350Skarels 136*16350Skarels send_delete() 137*16350Skarels { 138*16350Skarels /* tell the daemon to remove your invitation */ 139*16350Skarels 140*16350Skarels msg.type = DELETE; 141*16350Skarels 142*16350Skarels /* this is just a extra clean up, so just send it 143*16350Skarels and don't wait for an answer */ 144*16350Skarels 145*16350Skarels msg.id_num = remote_id; 146*16350Skarels daemon_addr.sin_addr = his_machine_addr; 147*16350Skarels if (sendto(ctl_sockt, &msg, sizeof(CTL_MSG), 0, &daemon_addr, 148*16350Skarels sizeof(daemon_addr)) != sizeof(CTL_MSG)) { 149*16350Skarels perror("send_delete remote"); 150*16350Skarels } 151*16350Skarels 152*16350Skarels msg.id_num = local_id; 153*16350Skarels daemon_addr.sin_addr = my_machine_addr; 154*16350Skarels if (sendto(ctl_sockt, &msg, sizeof(CTL_MSG), 0, &daemon_addr, 155*16350Skarels sizeof(daemon_addr)) != sizeof(CTL_MSG)) { 156*16350Skarels perror("send_delete local"); 157*16350Skarels } 158*16350Skarels } 159