-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathchatAppMultiCast.c
202 lines (170 loc) · 5.03 KB
/
chatAppMultiCast.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
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
/*******************************
References
*******************************/
// sending structs over sockets: https://stackoverflow.com/questions/4426016/sending-a-struct-with-char-pointer-inside
/*******************************
Error Codes
*******************************/
// 1: no error. returned succesfully
// -1: unknown error
// -2: connection error
// -4: invalid argument(s) passed
// TODO: set socket opt
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>// for inet_addr()
#include <arpa/inet.h> // for inet_addr()
#include <fcntl.h> // non-blocking
#include <unistd.h> // for close()
#include <string.h> // for strcmp()
#include <netdb.h> // getservbyname()
#include <pthread.h>
#include <errno.h>
// protocol struct
struct chat_proto{
char opcode;
unsigned char nameLength;
char name;
unsigned short textLength;
char message[1];
};
//your_program -mcip x.x.x.x. -port XX
struct args{
char ip_addr[20];
int port;
};
int MAX_BUFF = 65794;
int MAX_TEXT = 65536;
static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
struct args args;
struct sockaddr_in sock_addr;
struct ip_mreqn mreq;
char* name;
int sockfd;
char opcode = '1';
// sockfd, port#, ip/dns
int initApp();
void* localHandler(void *);
int cliListener(char**);
int setName();
int main(int argc, char *argv[]){
if(argc!=5){
printf("Error: Incorrect number of arguments: %d \nRequired 2 in the following order: \n-mcip x.x.x.x -port xx\n", argc-1);
return -4;
}
/*******************************
IP
*******************************/
const char ip_opt[] = "-mcip";
if(strcmp(ip_opt, argv[1])==0){
strcpy(args.ip_addr, argv[2]);
}
else{
printf("Invalid port argument.\n");
return -4;
}
/*******************************
PORT
*******************************/
const char port_opt[] = "-port";
if(strcmp(port_opt, argv[3])==0){
args.port = (int)atoi(argv[4]);
}
else{
printf("Invalid port argument.\n");
return -4;
}
sock_addr.sin_family = AF_INET;
sock_addr.sin_port = htons(args.port);
sock_addr.sin_addr.s_addr = htonl(INADDR_ANY);
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if(initApp()<0){
fprintf(stderr, "\nError: Application terminated.\n");
return -1;
}
return 1;
}
int initApp(){
pthread_attr_t attrs;
char loopch = 0;
pthread_attr_init(&attrs);
pthread_attr_setdetachstate(&attrs, PTHREAD_CREATE_DETACHED);
if(bind(sockfd, (struct sockaddr *) &sock_addr, sizeof(sock_addr)) < 0){
fprintf(stderr, "\nError: couldn't bind to the socket.\nInternal Error: %s\n", strerror(errno));
return -1;
}
printf("ip: %s, port: %d\n***\n", args.ip_addr, args.port);
mreq.imr_multiaddr.s_addr = inet_addr(args.ip_addr);
mreq.imr_address.s_addr = htonl(INADDR_ANY);
if(setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0){
fprintf(stderr, "\nError: couldn't add socket membership.\nInternal Error: %s\n", strerror(errno));
return -1;
}
if(setsockopt(sockfd, IPPROTO_IP, IP_MULTICAST_LOOP, (char *)&loopch, sizeof(loopch)) < 0){
fprintf(stderr, "Error: couldn't set IP_MULTICAST_LOOP.\nInternal Error: %s\n", strerror(errno));
//close(sd);
return -1;
}
if(setName()<0){
printf("Error: Couldn't set name.");
return -1;
}
// thread for handling cli input and sending
pthread_t t;
if(pthread_create(&t, &attrs, localHandler, 0)<0){
fprintf(stderr, "Error: couldn't create thread.\n Internal Error: %s\n", strerror(errno));
}
for(;;){
char recvBuff[65536] = {'\0'};
//mutex
if(strlen(recvBuff)!=0){
recvfrom(sockfd, (char*)&recvBuff, MAX_BUFF, 0, NULL, 0);
printf("message recieved: %s\n", recvBuff);
memset(recvBuff, 0, 65536);
}
}
return 1;
}
void* localHandler(void *thread_args){
char* input;
while(opcode == '1'){
cliListener(&input);
printf("input: %s\n", input);
createPacket(input);
}
// if opcode 2, send bye
//if anything else, send error and terminate
}
int createPacket(char* text){
char opcode = '1';
}
int cliListener(char** input){
char temp_input[65536] = {'\0'};
fgets(temp_input, MAX_TEXT, stdin);
int input_size = strlen(temp_input);
*input = malloc(input_size);
strcpy(*input, temp_input);
return 1;
}
int setName(){
char temp_input[255] = {'\0'};
while(temp_input[0]=='\0'){
printf("Enter a username for yourself: ");
fgets(temp_input, 255, stdin);
if(temp_input=='\0'){
printf("Name cant be empty. Try again.");
}
}
int input_size = strlen(temp_input);
name = malloc(input_size);
strcpy(name, temp_input);
printf("***Username Set***\n");
return 1;
}
/*******************************
Helper Functions
*******************************/