Mercurial > pt1.oyama
view recpt1/recpt1.c @ 17:52c7c7c64ba6
commented out debug messages.
author | Yoshiki Yazawa <yaz@honeyplanet.jp> |
---|---|
date | Wed, 25 Feb 2009 03:07:50 +0900 |
parents | ecb85bde67b1 |
children | 84ff6ef710ea |
line wrap: on
line source
#include <stdio.h> #include <fcntl.h> #include <sys/types.h> #include <sys/stat.h> #include <time.h> #include <stdlib.h> #include <string.h> #include <pthread.h> #include <math.h> #include <unistd.h> #include <getopt.h> #include <signal.h> #include <netdb.h> #include <arpa/inet.h> #include <netinet/in.h> #include <sys/ioctl.h> #include "pt1_ioctl.h" #include "recpt1.h" #include "decoder.h" /* globals */ int f_exit = FALSE; typedef struct sock_data { int sfd; /* socket fd */ struct sockaddr_in addr; } sock_data; typedef struct thread_data { QUEUE_T *queue; decoder *decoder; int wfd; /* output file fd */ sock_data *sock_data; } thread_data; /* lookup frequency conversion table*/ ISDB_T_FREQ_CONV_TABLE * searchrecoff(char *channel) { int lp; for(lp = 0; isdb_t_conv_table[lp].parm_freq != NULL; lp++) { /* return entry number in the table when strings match and * lengths are same. */ if((memcmp(isdb_t_conv_table[lp].parm_freq, channel, strlen(channel)) == 0) && (strlen(channel) == strlen(isdb_t_conv_table[lp].parm_freq))) { return &isdb_t_conv_table[lp]; } } return NULL; } QUEUE_T * create_queue(size_t size) { QUEUE_T *p_queue; int memsize = sizeof(QUEUE_T) + size * sizeof(BUFSZ); p_queue = (QUEUE_T*)calloc(memsize, sizeof(char)); if(p_queue != NULL) { p_queue->size = size; p_queue->no_full = size; p_queue->no_empty = 0; pthread_mutex_init(&p_queue->mutex, NULL); pthread_cond_init(&p_queue->cond_full, NULL); pthread_cond_init(&p_queue->cond_empty, NULL); } return p_queue; } void destroy_queue(QUEUE_T *p_queue) { if(!p_queue) return; pthread_mutex_destroy(&p_queue->mutex); pthread_cond_destroy(&p_queue->cond_full); pthread_cond_destroy(&p_queue->cond_empty); free(p_queue); } /* enqueue data. this function will block if queue is full. */ void enqueue(QUEUE_T *p_queue, BUFSZ *data) { pthread_mutex_lock(&p_queue->mutex); /* entered critical section */ /* wait until queue is not full */ while(!p_queue->no_full) { pthread_cond_wait(&p_queue->cond_full, &p_queue->mutex); fprintf(stderr, "Full\n"); } p_queue->buffer[p_queue->in] = data; p_queue->in++; p_queue->in %= p_queue->size; p_queue->no_full--; p_queue->no_empty++; /* leaving critical section */ pthread_mutex_unlock(&p_queue->mutex); pthread_cond_signal(&p_queue->cond_empty); } /* dequeue data. this function will block if queue is empty. */ BUFSZ * dequeue(QUEUE_T *p_queue) { BUFSZ *buffer; pthread_mutex_lock(&p_queue->mutex); /* entered the critical section*/ /* wait until queue is filled */ while (!p_queue->no_empty) { pthread_cond_wait(&p_queue->cond_empty, &p_queue->mutex); } /* take buffer address */ buffer = p_queue->buffer[p_queue->out]; /* move location marker to next position */ p_queue->out++; p_queue->out %= p_queue->size; /* update flags */ p_queue->no_full++; p_queue->no_empty--; /* leaving the critical section */ pthread_mutex_unlock(&p_queue->mutex); pthread_cond_signal(&p_queue->cond_full); return buffer; } /* this function will be reader thread */ void * write_func(void *p) { thread_data *data = (thread_data *)p; QUEUE_T *p_queue = data->queue; decoder *dec = data->decoder; int wfd = data->wfd; int use_b25 = dec ? TRUE : FALSE; int use_udp = data->sock_data ? TRUE : FALSE; int fileless = FALSE; int sfd = -1; struct sockaddr *addr = NULL; BUFSZ *buf; ARIB_STD_B25_BUFFER sbuf, dbuf; int code; if(wfd == -1) fileless = TRUE; if(use_udp) { sfd = data->sock_data->sfd; addr = (struct sockaddr *)&data->sock_data->addr; } while(1) { if(fileless) { buf = dequeue(p_queue); /* no entry in the queue */ if(buf == NULL) { break; } sbuf.data = buf->buffer; sbuf.size = buf->size; if(use_b25) { /* write data to output file*/ code = b25_decode(dec, &sbuf, &dbuf); if(code < 0) { fprintf(stderr, "b25_decode failed\n"); break; } if(use_udp && sfd != -1) { sendto(sfd, dbuf.data, dbuf.size, 0, addr, sizeof(struct sockaddr_in)); } free(buf); } else { if(use_udp && sfd != -1) { sendto(sfd, sbuf.data, sbuf.size, 0, addr, sizeof(struct sockaddr_in)); } free(buf); } /* normal exit */ if((f_exit) && (!p_queue->no_empty)) { if(use_b25) { code = b25_finish(dec, &sbuf, &dbuf); if(code < 0) { fprintf(stderr, "b25_finish failed\n"); close(sfd); break; } if(use_udp && sfd != -1) { sendto(sfd, dbuf.data, dbuf.size, 0, addr, sizeof(struct sockaddr_in)); } } close(sfd); break; } } /* fileless */ else { buf = dequeue(p_queue); /* no entry in the queue */ if(buf == NULL) { close(wfd); break; } sbuf.data = buf->buffer; sbuf.size = buf->size; if(use_b25) { /* write data to output file*/ code = b25_decode(dec, &sbuf, &dbuf); if(code < 0) { fprintf(stderr, "b25_decode failed\n"); close(wfd); break; } write(wfd, dbuf.data, dbuf.size); if(use_udp && sfd != -1) { sendto(sfd, dbuf.data, dbuf.size, 0, addr, sizeof(struct sockaddr_in)); } free(buf); } else { write(wfd, sbuf.data, sbuf.size); if(use_udp && sfd != -1) { sendto(sfd, sbuf.data, sbuf.size, 0, addr, sizeof(struct sockaddr_in)); } free(buf); } /* normal exit */ if((f_exit) && (!p_queue->no_empty)) { if(use_b25) { code = b25_finish(dec, &sbuf, &dbuf); if(code < 0) { fprintf(stderr, "b25_finish failed\n"); close(wfd); close(sfd); break; } write(wfd, dbuf.data, dbuf.size); if(use_udp && sfd != -1) { sendto(sfd, dbuf.data, dbuf.size, 0, addr, sizeof(struct sockaddr_in)); } } close(wfd); close(sfd); break; } } } return NULL; } void show_usage(char *cmd) { fprintf(stderr, "\n"); fprintf(stderr, "Usage: \n%s [--b25 [--round N] [--strip] [--EMM]] [--udp [--addr hostname --port portnumber]] channel recsec destfile\n", cmd); fprintf(stderr, "\n"); } void show_options(void) { fprintf(stderr, "Options:\n"); fprintf(stderr, "--b25: Decrypt using BCAS card\n"); fprintf(stderr, " --round N: Specify round number\n"); fprintf(stderr, " --strip: Strip null stream\n"); fprintf(stderr, " --EMM: Instruct EMM operation\n"); fprintf(stderr, "--udp: Turn on udp broadcasting\n"); fprintf(stderr, " --addr hostname: Hostname or address to connect\n"); fprintf(stderr, " --port portnumber: Port number to connect\n"); fprintf(stderr, "--help: Show this help\n"); fprintf(stderr, "\n"); } void show_channels(void) { fprintf(stderr, "Available Channels:\n"); fprintf(stderr, "13-52:Terrestrial Channels\n"); fprintf(stderr, "151ch:BS Asahi\n"); fprintf(stderr, "161ch:BS-i\n"); fprintf(stderr, "191ch:WOWOW\n"); fprintf(stderr, "171ch:BS Japan\n"); fprintf(stderr, "200ch:Star Channel\n"); fprintf(stderr, "211ch:BS11 Digital\n"); fprintf(stderr, "222ch:TwellV\n"); fprintf(stderr, "141ch:BS Nittele\n"); fprintf(stderr, "181ch:BS Fuji\n"); fprintf(stderr, "101ch:NHK BS1\n"); fprintf(stderr, "102ch:NHK BS2\n"); fprintf(stderr, "103ch:NHK BShi\n"); fprintf(stderr, "CS2-CS24:CS Channels\n"); fprintf(stderr, "\n"); } float getsignal_isdb_s(int signal) { /* apply linear interpolation */ static const float afLevelTable[] = { 24.07f, // 00 00 0 24.07dB 24.07f, // 10 00 4096 24.07dB 18.61f, // 20 00 8192 18.61dB 15.21f, // 30 00 12288 15.21dB 12.50f, // 40 00 16384 12.50dB 10.19f, // 50 00 20480 10.19dB 8.140f, // 60 00 24576 8.140dB 6.270f, // 70 00 28672 6.270dB 4.550f, // 80 00 32768 4.550dB 3.730f, // 88 00 34816 3.730dB 3.630f, // 88 FF 35071 3.630dB 2.940f, // 90 00 36864 2.940dB 1.420f, // A0 00 40960 1.420dB 0.000f // B0 00 45056 -0.01dB }; unsigned char sigbuf[4]; memset(sigbuf, '\0', sizeof(sigbuf)); sigbuf[0] = (((signal & 0xFF00) >> 8) & 0XFF); sigbuf[1] = (signal & 0xFF); /* calculate signal level */ if(sigbuf[0] <= 0x10U) { /* clipped maximum */ return 24.07f; } else if (sigbuf[0] >= 0xB0U) { /* clipped minimum */ return 0.0f; } else { /* linear interpolation */ const float fMixRate = (float)(((unsigned short)(sigbuf[0] & 0x0FU) << 8) | (unsigned short)sigbuf[0]) / 4096.0f; return afLevelTable[sigbuf[0] >> 4] * (1.0f - fMixRate) + afLevelTable[(sigbuf[0] >> 4) + 0x01U] * fMixRate; } } void calc_cn(int fd, int type) { int rc ; double P ; double CNR; if(ioctl(fd, GET_SIGNAL_STRENGTH, &rc) < 0) { printf("Tuner Select Error\n"); return ; } if(type == CHTYPE_GROUND) { P = log10(5505024/(double)rc) * 10; CNR = (0.000024 * P * P * P * P) - (0.0016 * P * P * P) + (0.0398 * P * P) + (0.5491 * P)+3.0965; printf("Signal=%fdB\n", CNR); } else { CNR = getsignal_isdb_s(rc); printf("Signal=%fdB\n", CNR); } } void cleanup(int tfd) { /* stop recording */ ioctl(tfd, STOP_REC, 0); /* restore LNB state */ #if 0 if(ptr->type == CHTYPE_SATELLITE) { if(ioctl(tfd, LNB_DISABLE, 0) < 0) { return 0 ; } } #endif /* xxx really need mutex? */ f_exit = TRUE; } /* will be signal handler thread */ void * process_signals(void *data) { sigset_t waitset; int sig; int tfd = *(int *)data; sigemptyset(&waitset); sigaddset(&waitset, SIGINT); sigaddset(&waitset, SIGTERM); sigwait(&waitset, &sig); switch(sig) { case SIGINT: fprintf(stderr, "\nSIGINT received. cleaning up...\n"); cleanup(tfd); break; case SIGTERM: fprintf(stderr, "\nSIGTERM received. cleaning up...\n"); cleanup(tfd); break; } return NULL; /* dummy */ } void init_signal_handlers(pthread_t *signal_thread, int tfd) { sigset_t blockset; static int tunerfd; tunerfd = tfd; sigemptyset(&blockset); sigaddset(&blockset, SIGINT); sigaddset(&blockset, SIGTERM); if(pthread_sigmask(SIG_BLOCK, &blockset, NULL)) fprintf(stderr, "pthread_sigmask() failed.\n"); pthread_create(signal_thread, NULL, process_signals, &tunerfd); } int main(int argc, char **argv) { int tfd, wfd; int lp; int recsec; int indefinite = FALSE; time_t start_time, cur_time; FREQUENCY freq; ISDB_T_FREQ_CONV_TABLE *ptr; pthread_t dequeue_thread; pthread_t signal_thread; QUEUE_T *p_queue = create_queue(MAX_QUEUE); BUFSZ *bufptr; decoder *dec = NULL; thread_data tdata; decoder_options dopt = { 4, /* round */ 0, /* strip */ 0 /* emm */ }; int result; int option_index; struct option long_options[] = { { "b25", 0, NULL, 'b'}, { "B25", 0, NULL, 'b'}, { "round", 1, NULL, 'r'}, { "strip", 0, NULL, 's'}, { "emm", 0, NULL, 'm'}, { "EMM", 0, NULL, 'm'}, { "udp", 0, NULL, 'u'}, { "addr", 1, NULL, 'a'}, { "port", 1, NULL, 'p'}, { "help", 0, NULL, 'h'}, {0, 0, 0, 0} /* terminate */ }; int use_b25 = FALSE; int use_udp = FALSE; int fileless = FALSE; char *host_to = NULL; int port_to = 1234; sock_data *sdata = NULL; while((result = getopt_long(argc, argv, "br:smua:p:h", long_options, &option_index)) != -1) { switch(result) { case 'b': use_b25 = TRUE; fprintf(stderr, "using B25...\n"); break; case 's': dopt.strip = TRUE; fprintf(stderr, "enable B25 strip\n"); break; case 'm': dopt.emm = TRUE; fprintf(stderr, "enable B25 emm processing\n"); break; case 'u': use_udp = TRUE; host_to = "localhost"; fprintf(stderr, "enable UDP broadcasting\n"); break; case 'h': show_usage(argv[0]); show_options(); show_channels(); exit(0); break; /* following options require argument */ case 'r': dopt.round = atoi(optarg); fprintf(stderr, "set round %d\n", dopt.round); break; case 'a': use_udp = TRUE; host_to = optarg; fprintf(stderr, "UDP destination address: %s\n", host_to); break; case 'p': port_to = atoi(optarg); fprintf(stderr, "UDP port: %d\n", port_to); break; } } if(argc - optind < 3) { if(argc - optind == 2 && use_udp) { fprintf(stderr, "Fileless UDP broadcasting\n"); fileless = TRUE; wfd = -1; } else { fprintf(stderr, "Arguments are necessary!\n"); fprintf(stderr, "Try '%s --help' for more information.\n", argv[0]); return 1; } } /* get channel */ ptr = searchrecoff(argv[optind]); if(ptr == NULL) { fprintf(stderr, "Channel Select Error(%s)\n", argv[optind]); return 1; } freq.frequencyno = ptr->set_freq; freq.slot = ptr->add_freq; if(ptr->type == CHTYPE_SATELLITE) { for(lp = 0; lp < 2; lp++) { tfd = open(bsdev[lp], O_RDONLY); if(tfd >= 0) { break; } } if(tfd < 0) { fprintf(stderr, "Cannot open tuner\n"); return 1; } } else { for(lp = 0; lp < 2; lp++) { tfd = open(isdb_t_dev[lp], O_RDONLY); if(tfd >= 0) { break; } } if(tfd < 0) { fprintf(stderr, "Cannot open tuner\n"); return 1; } } /* get recsec */ recsec = atoi(argv[optind + 1]); if(recsec <= 0) indefinite = TRUE; /* initialize decoder */ if(use_b25) { dec = b25_startup(&dopt); if(!dec) { fprintf(stderr, "Could not start b25 decoder\n"); fprintf(stderr, "Fall back to encrypted recording\n"); use_b25 = 0; } } /* initialize udp connection */ if(use_udp) { sdata = calloc(1, sizeof(sock_data)); struct in_addr ia; ia.s_addr = inet_addr(host_to); if(ia.s_addr == INADDR_NONE) { struct hostent *hoste = gethostbyname(host_to); if(!hoste) { perror("failed to get host by name"); return 1; } ia.s_addr = *(in_addr_t*) (hoste->h_addr_list[0]); } sdata->addr.sin_family = AF_INET; sdata->addr.sin_port = htons (port_to); sdata->addr.sin_addr.s_addr = ia.s_addr; if((sdata->sfd = socket(PF_INET, SOCK_DGRAM, 0)) < 0) { perror("socket"); return 1; } } /* open output file */ if(!fileless) { wfd = open(argv[optind + 2], (O_RDWR | O_CREAT | O_TRUNC), 0666); if(wfd < 0) { fprintf(stderr, "Could not open output file(%s)\n", argv[optind + 2]); return 1; } } if(ptr->type == CHTYPE_SATELLITE) { if(ioctl(tfd, LNB_ENABLE, 0) < 0) { return 0 ; } } if(ioctl(tfd, SET_CHANNEL, &freq) < 0) { fprintf(stderr, "Could not tune to the specified channel\n"); calc_cn(tfd, ptr->type); return 1; } calc_cn(tfd, ptr->type); /* init signal handler thread */ init_signal_handlers(&signal_thread, tfd); /* make reader thread */ tdata.queue = p_queue; tdata.decoder = dec; tdata.wfd = wfd; tdata.sock_data = sdata; pthread_create(&dequeue_thread, NULL, write_func, &tdata); /* start recording */ if(ioctl(tfd, START_REC, 0) < 0) { fprintf(stderr, "Tuner could not start recording\n"); return 1; } fprintf(stderr, "Recording...\n"); time(&start_time); /* read from tuner */ while(1) { if(f_exit) break; time(&cur_time); bufptr = malloc(sizeof(BUFSZ)); bufptr->size = read(tfd, bufptr->buffer, MAX_READ_SIZE); if(bufptr->size <= 0) { if((cur_time - start_time) >= recsec && !indefinite) { f_exit = TRUE; enqueue(p_queue, NULL); break; } else { continue; } } enqueue(p_queue, bufptr); /* stop recording */ if((cur_time - start_time) >= recsec && !indefinite) { ioctl(tfd, STOP_REC, 0); /* read remaining data */ while(1) { bufptr = malloc(sizeof(BUFSZ)); bufptr->size = read(tfd, bufptr->buffer, MAX_READ_SIZE); if(bufptr->size <= 0) { f_exit = TRUE; enqueue(p_queue, NULL); break; } enqueue(p_queue, bufptr); } break; } } /* close tuner */ if(ptr->type == CHTYPE_SATELLITE) { if(ioctl(tfd, LNB_DISABLE, 0) < 0) { return 0 ; } } close(tfd); /* wait reader thread */ pthread_join(dequeue_thread, NULL); // fprintf(stderr, "dequeue_thread joined\n"); pthread_join(signal_thread, NULL); // fprintf(stderr, "signal_thread joined\n"); /* relase queue */ destroy_queue(p_queue); /* free socket data */ free(sdata); /* release decoder */ if(use_b25) { b25_shutdown(dec); } // fprintf(stderr, "leaving main\n"); return 0; }