1 /*
2 * TI FM kernel driver's sample application.
3 *
4 * Copyright (C) 2010 Texas Instruments
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 *
19 */
20
21 #include <stdio.h>
22 #include <fcntl.h>
23 #include <linux/videodev2.h>
24 #include <math.h>
25 #include <pthread.h>
26 #include <errno.h>
27 #include <signal.h>
28 #include <string.h>
29 #include <stdlib.h>
30 #include <tinyalsa/asoundlib.h>
31 #include <poll.h>
32
33 #include "kfmapp.h"
34
35 static unsigned int pdevice = 0; /* playback device */
36 static unsigned int cdevice = 1; /* capture device */
37 static int fm_aud_enable;
38 struct pcm *pcm_p = NULL;
39 struct pcm *pcm_c = NULL;
40 struct mixer *mixer;
41
42 /* #define V4L2_TUNER_SUB_RDS 0x10 */
43
44 static char *g_mutemodes[]={"Mute ON","Mute OFF","Attenuate Voice"};
45 /*
46 static char *g_bands[]={"Europe/US","Japan"};
47 static char *g_sm_modes[]={"Stereo","Mono"};
48 static char *g_rx_deemphasis_modes[]={"50 usec","75 usec"};
49 static char *g_rds_opmodes[]={"RDS","RBDS"};
50 static char *g_af_switch_mode[]={"Off","On"};
51 */
52 static char *g_rds_modes[]={"Off","On"};
53 static int g_vol_to_set;
54 static pthread_t g_rds_thread_ptr;
55 volatile char g_rds_thread_terminate,g_rds_thread_running;
56
57 static int g_radio_fd;
58
59 /* Program Type */
60 static char *pty_str[]= {"None", "News", "Current Affairs",
61 "Information","Sport", "Education",
62 "Drama", "Culture","Science",
63 "Varied Speech", "Pop Music",
64 "Rock Music","Easy Listening",
65 "Light Classic Music", "Serious Classics",
66 "other Music","Weather", "Finance",
67 "Childrens Progs","Social Affairs",
68 "Religion", "Phone In", "Travel",
69 "Leisure & Hobby","Jazz", "Country",
70 "National Music","Oldies","Folk",
71 "Documentary", "Alarm Test", "Alarm"};
72
fmapp_display_tx_menu(void)73 void fmapp_display_tx_menu(void)
74 {
75 printf("Available FM TX Commands:\n");
76 printf("f <freq> tune to freq(in MHz)\n");
77 printf("gf get frequency(MHz)\n");
78 printf("e <val> set pre-emphasis filter value"
79 "(0 = OFF, 1 = 50 usec and 2 = 75 usec)\n");
80 /* printf("ge get pre-emphasis filter\n");*/
81 printf("p <val> set FM TX powerlevel (91 - 122)\n");
82 /* printf("gp get deemphasis filter\n");
83 printf("i <val> set FM TX antenna impedance value (0 = 50, 1 = 200 and 2 = 500)\n");
84 printf("gi get FM TX antenna impedance value\n");*/
85 printf("1 to set RDS Radio Text\n");
86 printf("2 to set RDS Radio PS Name\n");
87 printf("3 <value> to set RDS Radio PI code\n");
88 printf("4 <value> to set RDS Radio PTY\n");
89 printf("5 <AF Freq in KHz> to set RDS Radio Alternate Frequency\n");
90 }
fmapp_display_rx_menu(void)91 void fmapp_display_rx_menu(void)
92 {
93 printf("Available FM RX Commands:\n");
94 /* printf("p power on/off\n"); */
95 printf("f <freq> tune to freq(in MHz)\n");
96 printf("gf get frequency(MHz)\n");
97 printf("gr get rssi level\n");
98 printf("t turns RDS on/off\n");
99 printf("gt get RDS on/off\n");
100 printf("+ increases the volume\n");
101 printf("- decreases the volume\n");
102 printf("v <0-65535> sets the volume\n");
103 printf("gv get volume\n");
104 printf("b<value> switches Japan / Eur-Us (0=US/Eur & 1=Japan)\n");
105 printf("gb get band\n");
106 printf("s switches stereo / mono\n");
107 printf("gs get stereo/mono mode\n");
108 printf("m changes mute mode\n");
109 printf("gm get mute mode\n");
110 /* printf("e set deemphasis filter\n");
111 printf("ge get deemphasis filter\n");
112 printf("d set rf dependent mute\n");
113 printf("gd get rf dependent mute\n");
114 printf("z set rds system\n");
115 printf("gz get rds system\n"); */
116 printf("c<value> set rds af switch(0-OFF & 1=ON)\n");
117 printf("gc get rds af switch\n");
118 printf("< seek down\n");
119 printf("> seek up\n");
120 printf("? <(0)-(127)> set RSSI threshold\n");
121 printf("g? get rssi threshold\n");
122 printf("ga get tuner attributes\n");
123 /* printf("gn auto scan\n"); */
124 printf("A Start FM RX Audio Routing\n");
125 printf("q quit rx menu\n");
126 }
fmapp_get_tx_ant_imp(void)127 int fmapp_get_tx_ant_imp(void)
128 {
129 struct v4l2_control vctrl;
130 int res;
131
132 vctrl.id = V4L2_CID_TUNE_ANTENNA_CAPACITOR;
133
134 res = ioctl(g_radio_fd,VIDIOC_G_CTRL,&vctrl);
135 if(res < 0)
136 {
137 printf("Failed to get FM Tx antenna impedence value\n");
138 return res;
139 }
140
141 printf("FM Tx antenna impedence value is --> %d\n",vctrl.value);
142 return 0;
143 }
144
fmapp_get_tx_power_level(void)145 int fmapp_get_tx_power_level(void)
146 {
147 struct v4l2_control vctrl;
148 int res;
149
150 vctrl.id = V4L2_CID_TUNE_POWER_LEVEL;
151
152 res = ioctl(g_radio_fd,VIDIOC_G_CTRL,&vctrl);
153 if(res < 0)
154 {
155 printf("Failed to get FM Tx power level\n");
156 return res;
157 }
158
159 printf("FM Tx Power level is --> %d\n",vctrl.value);
160 return 0;
161 }
fmapp_get_premphasis_filter_mode(void)162 int fmapp_get_premphasis_filter_mode(void)
163 {
164 struct v4l2_control vctrl;
165 int res;
166
167 vctrl.id = V4L2_CID_TUNE_PREEMPHASIS;
168
169 res = ioctl(g_radio_fd,VIDIOC_G_CTRL,&vctrl);
170 if(res < 0)
171 {
172 printf("Failed to get preemphasis filter val\n");
173 return res;
174 }
175
176 printf("Preemphasis filter val is --> %d\n",vctrl.value);
177 return 0;
178 }
fmapp_get_tx_frequency(void)179 int fmapp_get_tx_frequency(void)
180 {
181 struct v4l2_frequency vf;
182 struct v4l2_modulator vm;
183 int res, div;
184
185 vm.index = 0;
186 res = ioctl(g_radio_fd, VIDIOC_G_MODULATOR, &vm);
187 if(res < 0)
188 {
189 printf("Failed to get modulator capabilities\n");
190 return res;
191 }
192
193 res = ioctl(g_radio_fd, VIDIOC_G_FREQUENCY,&vf);
194 if(res < 0)
195 {
196 printf("Failed to read current frequency\n");
197 return res;
198 }
199
200 div = (vm.capability & V4L2_TUNER_CAP_LOW) ? 1000 : 1;
201
202 printf("Transmitting at Frequency %3.2f MHz\n",vf.frequency /
203 ( 16000.0 * div));
204 return 0;
205 }
fmapp_get_rx_frequency(void)206 int fmapp_get_rx_frequency(void)
207 {
208 struct v4l2_frequency vf;
209 struct v4l2_tuner vt;
210 int res, div;
211
212 vt.index = 0;
213 res = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vt);
214 if(res < 0)
215 {
216 printf("Failed to get tuner capabilities\n");
217 return res;
218 }
219
220 res = ioctl(g_radio_fd, VIDIOC_G_FREQUENCY,&vf);
221 if(res < 0)
222 {
223 printf("Failed to read current frequency\n");
224 return res;
225 }
226
227 div = (vt.capability & V4L2_TUNER_CAP_LOW) ? 1000 : 1;
228
229 printf("Tuned to frequency %3.2f MHz \n",vf.frequency / ( 16.0 * div));
230 return 0;
231 }
232
fmapp_set_tx_rds_radio_text(void)233 int fmapp_set_tx_rds_radio_text(void)
234 {
235 struct v4l2_ext_controls_kfmapp vec;
236 struct v4l2_ext_control_kfmapp vctrls;
237 int res;
238 char rds_text[100];
239
240 vec.ctrl_class = V4L2_CTRL_CLASS_FM_TX;
241 vec.count = 1;
242 vctrls.id = V4L2_CID_RDS_TX_RADIO_TEXT;
243 printf("Enter RDS text to transmit\n");
244 scanf("%s", rds_text);
245 vctrls.string = rds_text;
246 vctrls.size = strlen(rds_text) + 1;
247 vec.controls = &vctrls;
248
249 printf("Entered RDS text is - %s and strlen = %d\n",vctrls.string, vctrls.size);
250 res = ioctl(g_radio_fd, VIDIOC_S_EXT_CTRLS, &vec);
251 if(res < 0)
252 {
253 printf("Failed to set FM Tx RDS Radio text\n");
254 return res;
255 }
256
257 printf("FM Modulator RDS Radio text is set and transmitted\n");
258
259 return res;
260 }
261
fmapp_set_tx_rds_radio_ps_name(void)262 int fmapp_set_tx_rds_radio_ps_name(void)
263 {
264 struct v4l2_ext_controls_kfmapp vec;
265 struct v4l2_ext_control_kfmapp vctrls;
266 int res;
267 char rds_text[100];
268
269 vec.ctrl_class = V4L2_CTRL_CLASS_FM_TX;
270 vec.count = 1;
271 vctrls.id = V4L2_CID_RDS_TX_PS_NAME;
272 printf("Enter RDS PS Name to transmit\n");
273 scanf("%s", rds_text);
274 vctrls.string = rds_text;
275 vctrls.size = strlen(rds_text) + 1;
276 vec.controls = &vctrls;
277
278 printf("Entered RDS text is - %s\n",vctrls.string);
279 res = ioctl(g_radio_fd, VIDIOC_S_EXT_CTRLS, &vec);
280 if(res < 0)
281 {
282 printf("Failed to set FM Tx RDS Radio PS Name\n");
283 return res;
284 }
285
286 printf("FM Modulator RDS Radio PS Name set and transmitted\n");
287
288 return res;
289 }
290
fmapp_set_tx_rds_radio_pi_code(char * cmd)291 int fmapp_set_tx_rds_radio_pi_code(char *cmd)
292 {
293 struct v4l2_ext_controls_kfmapp vec;
294 struct v4l2_ext_control_kfmapp vctrls;
295 int user_val;
296 int res;
297
298 sscanf(cmd, "%d", &user_val);
299
300 vec.ctrl_class = V4L2_CTRL_CLASS_FM_TX;
301 vec.count = 1;
302 vctrls.id = V4L2_CID_RDS_TX_PI;
303 vctrls.value = user_val;
304 vctrls.size = 0;
305 vec.controls = &vctrls;
306
307 res = ioctl(g_radio_fd, VIDIOC_S_EXT_CTRLS, &vec);
308 if(res < 0)
309 {
310 printf("Failed to set FM Tx RDS PI Code\n");
311 return res;
312 }
313
314 printf("Setting FM Tx RDS PI Code is Succesful\n");
315
316 return res;
317
318 }
319
fmapp_set_tx_rds_radio_af(char * cmd)320 int fmapp_set_tx_rds_radio_af(char *cmd)
321 {
322 int fd, res, af_freq;
323
324 fd = open(FMTX_RDS_AF_SYSFS_ENTRY, O_RDWR);
325 if (fd < 0) {
326 printf("Can't open %s", FMTX_RDS_AF_SYSFS_ENTRY);
327 return -1;
328 }
329
330 res = write(fd, cmd, FMAPP_AF_MAX_FREQ_RANGE);
331 if(res <= 0){
332 printf("Failed to set FM TX RDS Alternate Frequency\n");
333 goto exit;
334 }
335
336 printf("FM RDS Alternate Frequency is to %s Succesfully\n", cmd);
337 exit:
338 close(fd);
339 return res;
340
341 }
fmapp_set_tx_rds_radio_pty(char * cmd)342 int fmapp_set_tx_rds_radio_pty(char *cmd)
343 {
344 struct v4l2_ext_controls_kfmapp vec;
345 struct v4l2_ext_control_kfmapp vctrls;
346 int user_val;
347 int res;
348
349 sscanf(cmd, "%d", &user_val);
350
351 vec.ctrl_class = V4L2_CTRL_CLASS_FM_TX;
352 vec.count = 1;
353 vctrls.id = V4L2_CID_RDS_TX_PTY;
354 vctrls.value = user_val;
355 vctrls.size = 0;
356 vec.controls = &vctrls;
357
358 res = ioctl(g_radio_fd, VIDIOC_S_EXT_CTRLS, &vec);
359 if(res < 0)
360 {
361 printf("Failed to set FM Tx RDS PTY\n");
362 return res;
363 }
364
365 printf("Setting FM Tx RDS PTY is Succesful\n");
366
367 return res;
368
369 }
fmapp_set_tx_ant_imp(char * cmd)370 int fmapp_set_tx_ant_imp(char *cmd)
371 {
372 int user_val;
373 struct v4l2_control vctrl;
374 int res;
375
376 sscanf(cmd, "%d", &user_val);
377
378 vctrl.id = V4L2_CID_TUNE_ANTENNA_CAPACITOR;
379 vctrl.value = user_val;
380 res = ioctl(g_radio_fd,VIDIOC_S_CTRL,&vctrl);
381 if(res < 0)
382 {
383 printf("Failed to set FM Tx antenna impedence value\n");
384 return res;
385 }
386
387 printf("Setting FM Tx antenna impedence value to ---> %d\n",vctrl.value);
388 return 0;
389 }
390
fmapp_set_tx_power_level(char * cmd)391 int fmapp_set_tx_power_level(char *cmd)
392 {
393 struct v4l2_ext_controls_kfmapp vec;
394 struct v4l2_ext_control_kfmapp vctrls;
395 int user_val;
396 int res;
397
398 sscanf(cmd, "%d", &user_val);
399
400 vec.ctrl_class = V4L2_CTRL_CLASS_FM_TX;
401 vec.count = 1;
402 vctrls.id = V4L2_CID_TUNE_POWER_LEVEL;
403 vctrls.value = user_val;
404 vctrls.size = 0;
405 vec.controls = &vctrls;
406
407 res = ioctl(g_radio_fd, VIDIOC_S_EXT_CTRLS, &vec);
408 if(res < 0)
409 {
410 printf("Failed to set FM Tx power level\n");
411 return res;
412 }
413
414 printf("Setting FM Tx Power level to ---> %d\n", vctrls.value);
415
416 return res;
417
418 }
fmapp_set_premphasis_filter_mode(char * cmd)419 int fmapp_set_premphasis_filter_mode(char *cmd)
420 {
421 struct v4l2_ext_controls_kfmapp vec;
422 struct v4l2_ext_control_kfmapp vctrls;
423 int user_val;
424 int res;
425
426 sscanf(cmd, "%d", &user_val);
427
428 vec.ctrl_class = V4L2_CTRL_CLASS_FM_TX;
429 vec.count = 1;
430 vctrls.id = V4L2_CID_TUNE_PREEMPHASIS;
431 vctrls.value = user_val;
432 vctrls.size = 0;
433 vec.controls = &vctrls;
434
435 res = ioctl(g_radio_fd, VIDIOC_S_EXT_CTRLS, &vec);
436 if(res < 0)
437 {
438 printf("Failed to set preemphasis filter val\n");
439 return res;
440 }
441
442 printf("Setting preemphasis filter val success\n");
443
444 return res;
445
446 }
447
fmapp_set_tx_frequency(char * cmd)448 int fmapp_set_tx_frequency(char *cmd)
449 {
450 float user_freq;
451 struct v4l2_frequency vf;
452 struct v4l2_modulator vm;
453 int res, div;
454
455 sscanf(cmd, "%f", &user_freq);
456
457 vm.index = 0;
458 res = ioctl(g_radio_fd, VIDIOC_G_MODULATOR, &vm);
459 if(res < 0)
460 {
461 printf("Failed to get modulator capabilities\n");
462 return res;
463 }
464
465 vf.tuner = 0;
466 vf.frequency = rint(user_freq * 16000 + 0.5);
467
468 div = (vm.capability & V4L2_TUNER_CAP_LOW) ? 1000 : 1;
469 if (div == 1)
470 vf.frequency /= 1000;
471
472 res = ioctl(g_radio_fd, VIDIOC_S_FREQUENCY, &vf);
473 if(res < 0)
474 {
475 printf("Failed to set frequency %f\n",user_freq);
476 return res;
477 }
478 printf("Started Transmitting at %3.2f MHz Frequency\n", vf.frequency /
479 (16.0 * div));
480
481 return res;
482 }
fmapp_set_rx_frequency(char * cmd)483 int fmapp_set_rx_frequency(char *cmd)
484 {
485 float user_freq;
486 struct v4l2_frequency vf;
487 struct v4l2_tuner vt;
488 int res, div;
489
490 sscanf(cmd, "%f", &user_freq);
491
492 vf.tuner = 0;
493 /* As per V4L2 specifications VIDIOC_S_FREQUENCY ioctl expects tuning
494 * frequency in units of 62.5 KHz, or if the struct v4l2_tuner or struct
495 * v4l2_modulator capabilities flag V4L2_TUNER_CAP_LOW is set, in units
496 * of 62.5 Hz. But FM ST v4l2 driver presently handling the frequency in
497 * units of 1 KHz
498 */
499 vf.frequency = rint(user_freq * 16000 + 0.5);
500
501 vt.index = 0;
502 res = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vt);
503 if(res < 0)
504 {
505 printf("Failed to get tuner capabilities\n");
506 return res;
507 }
508
509 div = (vt.capability & V4L2_TUNER_CAP_LOW) ? 1000 : 1;
510 if (div == 1)
511 vf.frequency /= 1000;
512
513 if(vf.frequency < vt.rangelow || vf.frequency > vt.rangehigh){
514 printf("Failed to set frequency: Frequency is not in range"
515 "(%3.2f MHz to %3.2f MHz)\n", (vt.rangelow/(16.0 * div)),
516 (vt.rangehigh/(16.0 * div)));
517 return -EINVAL;
518 }
519
520 res = ioctl(g_radio_fd, VIDIOC_S_FREQUENCY, &vf);
521 if(res < 0)
522 {
523 printf("Failed to set frequency %f\n",user_freq);
524 return res;
525 }
526 printf("Tuned to frequency %3.2f MHz\n", vf.frequency / (16.0 * div));
527 return 0;
528 }
529
display_volume_bar(void)530 inline void display_volume_bar(void)
531 {
532 int index;
533 printf("\nVolume: ");
534 for(index=1; index<g_vol_to_set; index = index*1000)
535 printf("#");
536
537 printf("\nVolume is : %d\n",g_vol_to_set);
538
539 }
fmapp_set_rx_volume(char * cmd,int interactive,int vol_to_set)540 int fmapp_set_rx_volume(char *cmd,int interactive,int vol_to_set)
541 {
542 struct v4l2_control vctrl;
543 int res;
544
545 if(interactive == FMAPP_INTERACTIVE)
546 sscanf(cmd, "%d", &g_vol_to_set);
547 else
548 g_vol_to_set = vol_to_set;
549
550 vctrl.id = V4L2_CID_AUDIO_VOLUME;
551 vctrl.value = g_vol_to_set;
552 res = ioctl(g_radio_fd,VIDIOC_S_CTRL,&vctrl);
553 if(res < 0)
554 {
555 g_vol_to_set = 0;
556 printf("Failed to set volume\n");
557 return res;
558 }
559 printf("Setting volume to %d \n",g_vol_to_set);
560 return 0;
561 }
562
fmapp_get_rx_volume(void)563 int fmapp_get_rx_volume(void)
564 {
565 struct v4l2_control vctrl;
566 int res;
567
568 vctrl.id = V4L2_CID_AUDIO_VOLUME;
569 res = ioctl(g_radio_fd,VIDIOC_G_CTRL,&vctrl);
570 if(res < 0)
571 {
572 printf("Failed to get volume\n");
573 return res;
574 }
575 g_vol_to_set = vctrl.value;
576
577 printf("Radio Volume is set to %d\n",g_vol_to_set);
578 // display_volume_bar();
579 return 0;
580 }
581
fmapp_rx_increase_volume(void)582 int fmapp_rx_increase_volume(void)
583 {
584 int ret;
585
586 g_vol_to_set +=1;
587 if(g_vol_to_set > 70)
588 g_vol_to_set = 70;
589
590 ret = fmapp_set_rx_volume(NULL,FMAPP_BATCH,g_vol_to_set);
591 if(ret < 0)
592 return ret;
593
594 display_volume_bar();
595 return 0;
596 }
fmapp_rx_decrease_volume(void)597 int fmapp_rx_decrease_volume(void)
598 {
599 int ret;
600 g_vol_to_set -=1;
601 if(g_vol_to_set < 0)
602 g_vol_to_set = 0;
603
604 ret = fmapp_set_rx_volume(NULL,FMAPP_BATCH,g_vol_to_set);
605 if(ret < 0)
606 return ret;
607
608 display_volume_bar();
609 return 0;
610 }
fmapp_set_rx_mute_mode(void)611 int fmapp_set_rx_mute_mode(void)
612 {
613 struct v4l2_control vctrl;
614 static short int mute_mode = FM_MUTE_OFF;
615 int res;
616
617 vctrl.value = 0;
618 printf("Mutemode = %d\n",mute_mode);
619 switch (mute_mode)
620 {
621 case FM_MUTE_OFF:
622 mute_mode = FM_MUTE_ON;
623 break;
624
625 case FM_MUTE_ON:
626 mute_mode = FM_MUTE_OFF;
627 break;
628 }
629
630 vctrl.id = V4L2_CID_AUDIO_MUTE;
631 vctrl.value = mute_mode;
632 res = ioctl(g_radio_fd,VIDIOC_S_CTRL,&vctrl);
633 if(res < 0)
634 {
635 printf("Failed to set mute mode\n");
636 return res;
637 }
638
639 printf("Setting to \"%s\" \n",g_mutemodes[mute_mode]);
640 return 0;
641 }
fmapp_get_rx_mute_mode(void)642 int fmapp_get_rx_mute_mode(void)
643 {
644 struct v4l2_control vctrl;
645 int res;
646
647 vctrl.id = V4L2_CID_AUDIO_MUTE;
648 res = ioctl(g_radio_fd,VIDIOC_G_CTRL,&vctrl);
649 if(res < 0)
650 {
651 printf("Failed to get mute mode\n");
652 return res;
653 }
654
655 printf("%s\n",g_mutemodes[vctrl.value]);
656 return 0;
657 }
fmapp_rx_seek(int seek_direction)658 int fmapp_rx_seek(int seek_direction)
659 {
660 struct ti_v4l2_hw_freq_seek frq_seek;
661 int res;
662
663 printf("Seeking %s..\n",seek_direction?"up":"down");
664 frq_seek.type = 1;
665 frq_seek.seek_upward = seek_direction;
666 frq_seek.spacing = 200000;
667 frq_seek.wrap_around = 0;
668 errno = 0;
669 res = ioctl(g_radio_fd,VIDIOC_S_HW_FREQ_SEEK,&frq_seek);
670 if(errno == EAGAIN)
671 {
672 printf("Band limit reached\n");
673 }
674 else if(res <0)
675 {
676 printf("Seek operation failed\n");
677 return res;
678 }
679 /* Display seeked freq */
680 fmapp_get_rx_frequency();
681 return 0;
682 }
683
fmapp_set_rx_af_switch(char * cmd)684 int fmapp_set_rx_af_switch(char *cmd)
685 {
686 int fd, res;
687
688 fd = open(FMRX_RDS_AF_SYSFS_ENTRY, O_RDWR);
689 if (fd < 0) {
690 printf("Can't open %s", FMRX_RDS_AF_SYSFS_ENTRY);
691 return -1;
692 }
693
694 res = write(fd, cmd, sizeof(char));
695 if(res <= 0){
696 printf("Failed to set FM RDS AF Switch\n");
697 goto exit;
698 }
699
700 printf("FM RDS Alternate Frequency is %s\n",
701 atoi(cmd) == 0 ? "OFF":"ON");
702 exit:
703 close(fd);
704 return res;
705 }
706
fmapp_get_rx_af_switch(void)707 int fmapp_get_rx_af_switch(void)
708 {
709 unsigned char fm_rds_af;
710 int fd, res;
711
712 fd = open(FMRX_RDS_AF_SYSFS_ENTRY, O_RDONLY);
713 if (fd < 0) {
714 printf("Can't open %s", FMRX_RDS_AF_SYSFS_ENTRY);
715 return -1;
716 }
717
718 res = read(fd, &fm_rds_af, 1);
719 if(res < 0){
720 printf("reading %s failed %s\n",
721 FMRX_RDS_AF_SYSFS_ENTRY,strerror(res));
722 goto exit;
723 }
724
725 printf("FM RDS Alternate Frequency is %s \n",
726 (atoi((char *) &fm_rds_af)) == 0?"OFF":"ON");
727 exit:
728 close(fd);
729 return 0;
730 }
731
fmapp_get_rx_rssi_threshold(void)732 int fmapp_get_rx_rssi_threshold(void)
733 {
734 unsigned char fm_rssi_threshhold;
735 int fd, res;
736
737 fd = open(FMRX_RSSI_LVL_SYSFS_ENTRY, O_RDONLY);
738 if (fd < 0) {
739 printf("Can't open %s", FMRX_RSSI_LVL_SYSFS_ENTRY);
740 return -1;
741 }
742
743 res = read(fd, &fm_rssi_threshhold, 3);
744 if(res < 0){
745 printf("reading %s failed %s\n",
746 FMRX_RSSI_LVL_SYSFS_ENTRY,strerror(res));
747 goto exit;
748 }
749
750 printf("Current FM RSSI threshold level is %d \n",
751 atoi((char *) &fm_rssi_threshhold));
752
753 exit:
754 close(fd);
755 return res;
756 }
757
fmapp_set_rx_rssi_threshold(char * cmd)758 int fmapp_set_rx_rssi_threshold(char *cmd)
759 {
760 int fd, res;
761
762 fd = open(FMRX_RSSI_LVL_SYSFS_ENTRY, O_RDWR);
763 if (fd < 0) {
764 printf("Can't open %s", FMRX_RSSI_LVL_SYSFS_ENTRY);
765 return -1;
766 }
767
768 res = write(fd, cmd, sizeof(char) * 3);
769 if(res <= 0){
770 printf("Failed to set FM RSSI threshold level\n");
771 goto exit;
772 }
773
774 printf("FM RSSI threshold level is set to %d\n", atoi(cmd));
775
776 exit:
777 close(fd);
778 return res;
779 }
780
fmapp_set_band(char * cmd)781 int fmapp_set_band(char *cmd)
782 {
783 int fd, res;
784
785 fd = open(FMRX_BAND_SYSFS_ENTRY, O_RDWR);
786 if (fd < 0) {
787 printf("Can't open %s", FMRX_BAND_SYSFS_ENTRY);
788 return -1;
789 }
790
791 res = write(fd, cmd, sizeof(char));
792 if(res <= 0){
793 printf("Failed to set FM Band\n");
794 goto exit;
795 }
796
797 printf("FM Band is set to %s\n", atoi(cmd) == 0?"US/EUROPE":"JAPAN");
798 exit:
799 close(fd);
800 return res;
801 }
802
fmapp_get_band(void)803 int fmapp_get_band(void)
804 {
805 unsigned char fm_band;
806 int fd, res;
807
808 fd = open(FMRX_BAND_SYSFS_ENTRY, O_RDONLY);
809 if (fd < 0) {
810 printf("Can't open %s", FMRX_BAND_SYSFS_ENTRY);
811 return -1;
812 }
813
814 res = read(fd, &fm_band, 1);
815 if(res < 0){
816 printf("reading %s failed %s\n",FMRX_BAND_SYSFS_ENTRY,strerror(res));
817 goto exit;
818 }
819
820 printf("Present FM Band is %s \n",
821 (atoi((char *) &fm_band)) == 0?"US/EUROPE":"JAPAN");
822 exit:
823 close(fd);
824 return res;
825 }
tinymix_set_value(struct mixer * mixer,unsigned int id,int value)826 static void tinymix_set_value(struct mixer *mixer, unsigned int id,
827 int value)
828 {
829 struct mixer_ctl *ctl;
830 enum mixer_ctl_type type;
831 unsigned int i, num_values;
832
833 ctl = mixer_get_ctl(mixer, id);
834 type = mixer_ctl_get_type(ctl);
835 num_values = mixer_ctl_get_num_values(ctl);
836
837 for(i=0; i<num_values; i++) {
838 if (mixer_ctl_set_value(ctl, i, value)) {
839 fprintf(stderr, "Error: invalid value\n");
840 return;
841 }
842 }
843 }
844
fmapp_start_audio()845 int fmapp_start_audio()
846 {
847 struct pcm_config config;
848
849 mixer = mixer_open(0);
850 if (!mixer) {
851 fprintf(stderr, "Failed to open mixer\n");
852 return EXIT_FAILURE;
853 }
854
855 config.channels = 2;
856 config.rate = 48000;
857 config.period_size = 1024;
858 config.period_count = 4;
859 config.format = PCM_FORMAT_S16_LE;
860 config.silence_threshold = 0;
861 config.stop_threshold = -1;
862
863 if (fm_aud_enable == 0){
864 /* Set Tinymix controles */
865 tinymix_set_value(mixer, 77, 2);
866 tinymix_set_value(mixer, 76, 2);
867 tinymix_set_value(mixer, 64, 1);
868 tinymix_set_value(mixer, 65, 4);
869 tinymix_set_value(mixer, 55, 12);
870 tinymix_set_value(mixer, 54, 11);
871 tinymix_set_value(mixer, 51, 1);
872 tinymix_set_value(mixer, 9, 120);
873 tinymix_set_value(mixer, 72, 1);
874 tinymix_set_value(mixer, 73, 1);
875 tinymix_set_value(mixer, 34, 1);
876 tinymix_set_value(mixer, 50, 1);
877
878 pcm_p = pcm_open(0, pdevice, PCM_OUT, &config);
879 if (!pcm_p || !pcm_is_ready(pcm_p)) {
880 fprintf(stderr, "Unable to open PCM device (%s)\n",
881 pcm_get_error(pcm_p));
882 return 0;
883 }
884 printf("Playback device opened successfully");
885 pcm_c = pcm_open(0, cdevice, PCM_IN, &config);
886 if (!pcm_c || !pcm_is_ready(pcm_c)) {
887 fprintf(stderr, "Unable to open PCM device (%s)\n",
888 pcm_get_error(pcm_c));
889 return 0;
890 }
891 printf("Capture device opened successfully");
892 pcm_start(pcm_c);
893 pcm_start(pcm_p);
894 printf(" Trigered the loopback");
895 fm_aud_enable = 1;
896 }
897 else {
898 /* Set Tinymix controls to Normal*/
899 tinymix_set_value(mixer, 77, 0);
900 tinymix_set_value(mixer, 76, 0);
901 tinymix_set_value(mixer, 64, 0);
902 tinymix_set_value(mixer, 65, 0);
903 tinymix_set_value(mixer, 55, 0);
904 tinymix_set_value(mixer, 54, 0);
905 tinymix_set_value(mixer, 51, 0);
906 tinymix_set_value(mixer, 9, 0);
907 tinymix_set_value(mixer, 72, 0);
908 tinymix_set_value(mixer, 73, 0);
909 tinymix_set_value(mixer, 34, 0);
910 tinymix_set_value(mixer, 50, 0);
911
912 /* close the device */
913 pcm_stop(pcm_p);
914 pcm_stop(pcm_c);
915 pcm_close(pcm_p);
916 pcm_close(pcm_c);
917 fm_aud_enable = 0;
918 }
919 printf("FM RX Audio Routing Done\n");
920 return 0;
921 }
922
fmapp_get_rx_rssi_lvl(void)923 int fmapp_get_rx_rssi_lvl(void)
924 {
925 struct v4l2_tuner vtun;
926 float rssi_lvl;
927 int res;
928
929 vtun.index = 0;
930 res = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vtun);
931 if(res < 0)
932 {
933 printf("Failed to get tunner attributes\n");
934 return res;
935 }
936 rssi_lvl = ((float)vtun.signal / 0xFFFF) * 100;
937 printf("Signal Strength: %d%%\n",(unsigned int)rssi_lvl);
938
939 return 0;
940 }
fmapp_set_stereo_mono_mode(void)941 int fmapp_set_stereo_mono_mode(void)
942 {
943 struct v4l2_tuner vtun;
944 int res = 0;
945
946 vtun.index = 0;
947 res = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vtun);
948 if(res < 0)
949 {
950 printf("Failed to set stereo-mono mode\n");
951 return res;
952 }
953
954 if(V4L2_TUNER_MODE_STEREO == vtun.audmode)
955 vtun.audmode = V4L2_TUNER_MODE_MONO;
956 else
957 vtun.audmode = V4L2_TUNER_MODE_STEREO;
958
959 res = ioctl(g_radio_fd, VIDIOC_S_TUNER, &vtun);
960 if(res < 0)
961 {
962 printf("Failed to set stereo-mono mode\n");
963 return res;
964 }
965 printf("Audio Mode set to: %s\n",(vtun.audmode == V4L2_TUNER_MODE_STEREO) ? "STEREO":"MONO");
966
967 return 0;
968 }
fmapp_get_stereo_mono_mode(void)969 int fmapp_get_stereo_mono_mode(void)
970 {
971 struct v4l2_tuner vtun;
972 int res;
973
974 vtun.index = 0;
975 res = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vtun);
976 if(res < 0)
977 {
978 printf("Failed to get tunner attributes\n");
979 return res;
980 }
981 printf("Audio Mode: %s\n",(vtun.audmode == V4L2_TUNER_MODE_STEREO) ? "STEREO":"MONO");
982
983 return 0;
984 }
fmapp_get_rx_tunner_attributes(void)985 int fmapp_get_rx_tunner_attributes(void)
986 {
987 struct v4l2_tuner vtun;
988 float sigstrength_percentage;
989 int res;
990
991 vtun.index = 0;
992 res = ioctl(g_radio_fd,VIDIOC_G_TUNER,&vtun);
993 if(res < 0)
994 {
995 printf("Failed to get tunner attributes\n");
996 return res;
997 }
998 printf("-----------------------\n");
999 printf("Tuner Name: %s\n",vtun.name);
1000 /* TODO: FM driver is not setting V4L2_TUNER_CAP_LOW flag , but its returning vtun.rangelow
1001 * and vtun.rangehigh ranges in HZ . This needs to be corrected in FM driver */
1002 printf(" Low Freq: %d KHz\n",
1003 (unsigned int )((float)vtun.rangelow * 0.0625));
1004 printf(" High Freq: %d KHz\n",
1005 (unsigned int) ((float)vtun.rangehigh * 0.0625));
1006 printf("Audio Mode: %s\n",(vtun.audmode == V4L2_TUNER_MODE_STEREO) ? "STEREO":"MONO");
1007 sigstrength_percentage = ((float)vtun.signal /0xFFFF) * 100;
1008 printf("Signal Strength: %d%%\n",(unsigned int)sigstrength_percentage);
1009 printf("-----------------------\n");
1010 return 0;
1011 }
1012
fmapp_get_scan_valid_frequencies(void)1013 int fmapp_get_scan_valid_frequencies(void)
1014 {
1015 int ret;
1016 struct v4l2_tuner vtun;
1017 struct v4l2_frequency vf;
1018 struct v4l2_control vctrl;
1019 float freq_multiplicator,start_frq,end_frq,
1020 freq,perc,threshold,divide_by;
1021 long totsig;
1022 unsigned char index;
1023
1024 vtun.index = 0;
1025 ret = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vtun); /* get frequency range */
1026 if (ret < 0) {
1027 printf("Failed to get frequency range");
1028 return ret;
1029 }
1030 freq_multiplicator = (62.5 * ((vtun.capability & V4L2_TUNER_CAP_LOW)
1031 ? 1 : 1000));
1032
1033 divide_by = (vtun.capability & V4L2_TUNER_CAP_LOW) ? 1000000 : 1000;
1034 start_frq = ((float)vtun.rangelow * freq_multiplicator)/divide_by;
1035 end_frq = ((float)vtun.rangehigh * freq_multiplicator)/divide_by;
1036
1037 threshold = FMAPP_ASCAN_SIGNAL_THRESHOLD_PER;
1038
1039 /* Enable Mute */
1040 vctrl.id = V4L2_CID_AUDIO_MUTE;
1041 vctrl.value = FM_MUTE_ON;
1042 ret = ioctl(g_radio_fd,VIDIOC_S_CTRL,&vctrl);
1043 if(ret < 0)
1044 {
1045 printf("Failed to set mute mode\n");
1046 return ret;
1047 }
1048 printf("Auto Scanning..\n");
1049 for(freq=start_frq;freq<=end_frq;freq+=0.1)
1050 {
1051 vf.tuner = 0;
1052 vf.frequency = rint(freq*1000);
1053 ret = ioctl(g_radio_fd, VIDIOC_S_FREQUENCY, &vf); /* tune */
1054 if (ret < 0) {
1055 printf("failed to set freq");
1056 return ret;
1057 }
1058 totsig = 0;
1059 for(index=0;index<FMAPP_ASCAN_NO_OF_SIGNAL_SAMPLE;index++)
1060 {
1061 vtun.index = 0;
1062 ret = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vtun); /* get info */
1063 if (ret < 0) {
1064 printf("Failed to get frequency range");
1065 return ret;
1066 }
1067 totsig += vtun.signal;
1068 perc = (totsig / (65535.0 * index));
1069 usleep(1);
1070 }
1071 perc = (totsig / (65535.0 * FMAPP_ASCAN_NO_OF_SIGNAL_SAMPLE));
1072 if ((perc*100.0) > threshold)
1073 printf("%2.1f MHz(%d%%)\n",freq,((unsigned short)(perc * 100.0)));
1074 }
1075 /* Disable Mute */
1076 vctrl.id = V4L2_CID_AUDIO_MUTE;
1077 vctrl.value = FM_MUTE_OFF;
1078 ret = ioctl(g_radio_fd,VIDIOC_S_CTRL,&vctrl);
1079 if(ret < 0)
1080 {
1081 printf("Failed to set mute mode\n");
1082 return ret;
1083 }
1084 printf("Scan Completed\n");
1085 return 0;
1086 }
fmapp_get_rds_onoff(void)1087 int fmapp_get_rds_onoff(void)
1088 {
1089 struct v4l2_tuner vtun;
1090 int res = 0;
1091
1092 vtun.index = 0;
1093 res = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vtun);
1094 if(res < 0)
1095 {
1096 printf("Failed to read RDS state\n");
1097 return res;
1098 }
1099 printf("RDS is: %s\n",(vtun.rxsubchans & V4L2_TUNER_SUB_RDS) ? "ON":"OFF");
1100
1101 return 0;
1102 }
fmapp_rds_decode(int blkno,int byte1,int byte2)1103 void fmapp_rds_decode(int blkno, int byte1, int byte2)
1104 {
1105 static char rds_psn[9];
1106 static char rds_txt[65];
1107 static int rds_pty,ms_code;
1108 static int group,spare,blkc_byte1,blkc_byte2;
1109
1110 switch (blkno) {
1111 case 0: /* Block A */
1112 printf("----------------------------------------\n");
1113 printf("block A - id=%d\n",(byte1 << 8) | byte2);
1114 break;
1115 case 1: /* Block B */
1116 printf("block B - group=%d%c tp=%d pty=%d spare=%d\n",
1117 (byte1 >> 4) & 0x0f,
1118 ((byte1 >> 3) & 0x01) + 'A',
1119 (byte1 >> 2) & 0x01,
1120 ((byte1 << 3) & 0x18) | ((byte2 >> 5) & 0x07),
1121 byte2 & 0x1f);
1122 group = (byte1 >> 3) & 0x1f;
1123 spare = byte2 & 0x1f;
1124 rds_pty = ((byte1 << 3) & 0x18) | ((byte2 >> 5) & 0x07);
1125 ms_code = (byte2 >> 3)& 0x1;
1126 break;
1127 case 2: /* Block C */
1128 printf("block C - 0x%02x 0x%02x\n",byte1,byte2);
1129 blkc_byte1 = byte1;
1130 blkc_byte2 = byte2;
1131 break;
1132 case 3 : /* Block D */
1133 printf("block D - 0x%02x 0x%02x\n",byte1,byte2);
1134 switch (group) {
1135 case 0: /* Group 0A */
1136 rds_psn[2*(spare & 0x03)+0] = byte1;
1137 rds_psn[2*(spare & 0x03)+1] = byte2;
1138 if ((spare & 0x03) == 0x03)
1139 printf("PSN: %s, PTY: %s, MS: %s\n",rds_psn,
1140 pty_str[rds_pty],ms_code?"Music":"Speech");
1141 break;
1142 case 4: /* Group 2A */
1143 rds_txt[4*(spare & 0x0f)+0] = blkc_byte1;
1144 rds_txt[4*(spare & 0x0f)+1] = blkc_byte2;
1145 rds_txt[4*(spare & 0x0f)+2] = byte1;
1146 rds_txt[4*(spare & 0x0f)+3] = byte2;
1147 /* Display radio text once we get 16 characters */
1148 // if ((spare & 0x0f) == 0x0f)
1149 if (spare > 16)
1150 {
1151 printf("Radio Text: %s\n",rds_txt);
1152 // memset(&rds_txt,0,sizeof(rds_txt));
1153 }
1154 break;
1155 }
1156 printf("----------------------------------------\n");
1157 break;
1158 default:
1159 printf("unknown block [%d]\n",blkno);
1160 }
1161 }
rds_thread(void * data)1162 void *rds_thread(void *data)
1163 {
1164 unsigned char buf[600];
1165 int radio_fd;
1166 int ret,index;
1167 struct pollfd pfd;
1168
1169 radio_fd = (int)data;
1170
1171 while(!g_rds_thread_terminate)
1172 {
1173 while(1){
1174 memset(&pfd, 0, sizeof(pfd));
1175 pfd.fd = radio_fd;
1176 pfd.events = POLLIN;
1177 ret = poll(&pfd, 1, 10);
1178 if (ret == 0){
1179 /* Break the poll after RDS data available */
1180 break;
1181 }
1182 }
1183
1184 ret = read(radio_fd,buf,500);
1185 if(ret < 0) {
1186
1187 break;
1188 }
1189 else if( ret > 0)
1190 {
1191 for(index=0;index<ret;index+=3)
1192 fmapp_rds_decode(buf[index+2] & 0x7,buf[index+1],buf[index]);
1193 }
1194 }
1195 /* TODO: Need to conform thread termination.
1196 * below msg is not coming ,have a doubt on thread termination.
1197 * Fix this later. */
1198 printf("RDS thread exiting..\n");
1199 return NULL;
1200 }
fmapp_set_rds_onoff(unsigned char fmapp_mode)1201 int fmapp_set_rds_onoff(unsigned char fmapp_mode)
1202 {
1203 struct v4l2_tuner vtun;
1204 int ret;
1205 static unsigned char rds_mode = FM_RDS_DISABLE;
1206
1207 vtun.index = 0;
1208 ret = ioctl(g_radio_fd, VIDIOC_G_TUNER, &vtun);
1209 if(ret < 0)
1210 {
1211 printf("Failed to get tuner capabilities\n");
1212 return ret;
1213 }
1214 if(rds_mode == FM_RDS_DISABLE) {
1215 vtun.rxsubchans |= V4L2_TUNER_SUB_RDS;
1216 rds_mode = FM_RDS_ENABLE;
1217 } else {
1218 vtun.rxsubchans &= ~V4L2_TUNER_SUB_RDS;
1219 rds_mode = FM_RDS_DISABLE;
1220 }
1221
1222 ret = ioctl(g_radio_fd, VIDIOC_S_TUNER, &vtun);
1223 if(ret < 0)
1224 {
1225 printf("Failed to set rds on/off status\n");
1226 return ret;
1227 }
1228 /* Create rds receive thread once */
1229 if(fmapp_mode == FM_MODE_RX && rds_mode == FM_RDS_ENABLE &&
1230 g_rds_thread_running == 0)
1231 {
1232 g_rds_thread_running = 1;
1233 pthread_create(&g_rds_thread_ptr,NULL,rds_thread,(void *)g_radio_fd);
1234 }
1235
1236 printf("RDS %s\n",g_rds_modes[rds_mode]);
1237 return 0;
1238 }
1239
fmapp_execute_tx_get_command(char * cmd)1240 void fmapp_execute_tx_get_command(char *cmd)
1241 {
1242 switch(cmd[0])
1243 {
1244 case 'f':
1245 fmapp_get_tx_frequency();
1246 break;
1247 case 'e':
1248 fmapp_get_premphasis_filter_mode();
1249 break;
1250 case 'p':
1251 fmapp_get_tx_power_level();
1252 break;
1253 case 'i':
1254 fmapp_get_tx_ant_imp();
1255 break;
1256 default:
1257 printf("unknown command; type 'h' for help\n");
1258 }
1259
1260 }
fmapp_execute_rx_get_command(char * cmd)1261 void fmapp_execute_rx_get_command(char *cmd)
1262 {
1263 switch(cmd[0])
1264 {
1265 case 'f':
1266 fmapp_get_rx_frequency();
1267 break;
1268 case 'r':
1269 fmapp_get_rx_rssi_lvl();
1270 break;
1271 case 't':
1272 fmapp_get_rds_onoff();
1273 break;
1274 case 'v':
1275 fmapp_get_rx_volume();
1276 break;
1277 case 'm':
1278 fmapp_get_rx_mute_mode();
1279 break;
1280 case 'b':
1281 fmapp_get_band();
1282 break;
1283 case 'c':
1284 fmapp_get_rx_af_switch();
1285 break;
1286 case '?':
1287 fmapp_get_rx_rssi_threshold();
1288 break;
1289 #if 0
1290 case 'd':
1291 fmapp_get_rfmute(fm_snd_ctrl);
1292 break;
1293 case 'z':
1294 fmapp_get_rds_operation_mode(fm_snd_ctrl);
1295 break;
1296 #endif
1297 case 's':
1298 fmapp_get_stereo_mono_mode();
1299 break;
1300 #if 0
1301 case 'e':
1302 fmapp_get_rx_deemphasis_filter_mode(fm_snd_ctrl);
1303 break;
1304 #endif
1305 case 'a':
1306 fmapp_get_rx_tunner_attributes();
1307 break;
1308 #if 0
1309 case 'n':
1310 fmapp_get_scan_valid_frequencies();
1311 break;
1312 #endif
1313 default:
1314 printf("unknown command; type 'h' for help\n");
1315 }
1316 }
fmapp_execute_rx_other_command(char * cmd)1317 void fmapp_execute_rx_other_command(char *cmd)
1318 {
1319 switch(cmd[0])
1320 {
1321 #if 0
1322 case 'p':
1323 fmapp_change_rx_power_mode(fm_snd_ctrl);
1324 break;
1325 #endif
1326 case 'f':
1327 fmapp_set_rx_frequency(cmd+1);
1328 break;
1329 case 't':
1330 fmapp_set_rds_onoff(FM_MODE_RX);
1331 break;
1332 case '+':
1333 fmapp_rx_increase_volume();
1334 break;
1335 case '-':
1336 fmapp_rx_decrease_volume();
1337 break;
1338 case 'v':
1339 fmapp_set_rx_volume(cmd+1,FMAPP_INTERACTIVE,0);
1340 break;
1341 case 'm':
1342 fmapp_set_rx_mute_mode();
1343 break;
1344 case '<':
1345 fmapp_rx_seek(FM_SEARCH_DIRECTION_DOWN);
1346 break;
1347 case '>':
1348 fmapp_rx_seek(FM_SEARCH_DIRECTION_UP);
1349 break;
1350 case 'b':
1351 fmapp_set_band(cmd+1);
1352 break;
1353 case 'h':
1354 fmapp_display_rx_menu();
1355 break;
1356 case 'c':
1357 fmapp_set_rx_af_switch(cmd+1);
1358 break;
1359 case '?':
1360 fmapp_set_rx_rssi_threshold(cmd+1);
1361 break;
1362 #if 0
1363 case 'd':
1364 fmapp_set_rfmute(fm_snd_ctrl);
1365 break;
1366 case 'z':
1367 fmapp_set_rds_operation_mode(fm_snd_ctrl);
1368 break;
1369 #endif
1370 case 's':
1371 fmapp_set_stereo_mono_mode();
1372 break;
1373 #if 0
1374 case 'e':
1375 fmapp_set_rx_deemphasis_filter_mode(fm_snd_ctrl);
1376 break;
1377 #endif
1378 case 'A':
1379 fmapp_start_audio();
1380 break;
1381 }
1382 }
1383
fmapp_execute_tx_other_command(char * cmd)1384 void fmapp_execute_tx_other_command(char *cmd)
1385 {
1386 switch(cmd[0])
1387 {
1388 case 'f':
1389 fmapp_set_tx_frequency(cmd+1);
1390 break;
1391 case 'e':
1392 fmapp_set_premphasis_filter_mode(cmd+1);
1393 break;
1394 case 'p':
1395 fmapp_set_tx_power_level(cmd+1);
1396 break;
1397 case 'i':
1398 fmapp_set_tx_ant_imp(cmd+1);
1399 break;
1400 case '1':
1401 fmapp_set_tx_rds_radio_text();
1402 break;
1403 case '2':
1404 fmapp_set_tx_rds_radio_ps_name();
1405 break;
1406 case '3':
1407 fmapp_set_tx_rds_radio_pi_code(cmd+1);
1408 break;
1409 case '4':
1410 fmapp_set_tx_rds_radio_pty(cmd+1);
1411 break;
1412 case '5':
1413 fmapp_set_tx_rds_radio_af(cmd+1);
1414 break;
1415 case 'h':
1416 fmapp_display_tx_menu();
1417 break;
1418 }
1419 }
1420 /* Switch to RX mode before accepting user commands for RX */
fmapp_execute_rx_command(void)1421 void fmapp_execute_rx_command(void)
1422 {
1423 char cmd[100];
1424 struct v4l2_tuner vtun;
1425 int ret;
1426
1427 vtun.index = 0;
1428 vtun.audmode = V4L2_TUNER_MODE_STEREO;
1429 vtun.rxsubchans = V4L2_TUNER_SUB_RDS;
1430 ret = ioctl(g_radio_fd, VIDIOC_S_TUNER, &vtun);
1431 if(ret < 0)
1432 {
1433 printf("Failed to set RX mode\n");
1434 return;
1435 }
1436
1437 printf("Switched to RX menu\n");
1438 printf("type 'h' for help\n");
1439
1440 while(1)
1441 {
1442 fgets(cmd, sizeof(cmd), stdin);
1443 switch(cmd[0]) {
1444 case 'g':
1445 fmapp_execute_rx_get_command(cmd+1);
1446 break;
1447 case 'q':
1448 printf("quiting RX menu\n");
1449 if (pcm_p != NULL && pcm_c != NULL)
1450 fmapp_start_audio();
1451 return;
1452 default:
1453 fmapp_execute_rx_other_command(cmd);
1454 break;
1455 }
1456 }
1457 }
fmapp_execute_tx_command(void)1458 void fmapp_execute_tx_command(void)
1459 {
1460 char cmd[100];
1461 struct v4l2_modulator vmod;
1462 int ret;
1463
1464 vmod.index = 0;
1465 vmod.txsubchans = V4L2_TUNER_SUB_STEREO | V4L2_TUNER_SUB_RDS;
1466
1467 ret = ioctl(g_radio_fd, VIDIOC_S_MODULATOR, &vmod);
1468 if(ret < 0)
1469 {
1470 printf("Failed to set TX mode\n");
1471 return;
1472 }
1473
1474 printf("Switched to TX menu\n");
1475 printf("type 'h' for help\n");
1476
1477 while(1)
1478 {
1479 fgets(cmd, sizeof(cmd), stdin);
1480 switch(cmd[0]) {
1481 case 'g':
1482 fmapp_execute_tx_get_command(cmd+1);
1483 break;
1484 case 'q':
1485 printf("quiting TX menu\n");
1486 return;
1487 default:
1488 fmapp_execute_tx_other_command(cmd);
1489 break;
1490 }
1491 }
1492 }
1493
fmapp_read_anddisplay_capabilities(void)1494 int fmapp_read_anddisplay_capabilities(void)
1495 {
1496 struct v4l2_capability cap;
1497 int res;
1498
1499 res = ioctl(g_radio_fd,VIDIOC_QUERYCAP,&cap);
1500 if(res < 0)
1501 {
1502 printf("Failed to read %s capabilities\n",DEFAULT_RADIO_DEVICE);
1503 return res;
1504 }
1505 if((cap.capabilities & V4L2_CAP_RADIO) == 0)
1506 {
1507 printf("%s is not radio devcie",DEFAULT_RADIO_DEVICE);
1508 return -1;
1509 }
1510 printf("\n***%s Info ****\n",DEFAULT_RADIO_DEVICE);
1511 printf("Driver : %s\n",cap.driver);
1512 printf("Card : %s\n",cap.card);
1513 printf("Bus : %s\n",cap.bus_info);
1514 printf("Capabilities : 0x%x\n",cap.capabilities);
1515
1516 return 0;
1517 }
1518
sig_handler()1519 static void sig_handler()
1520 {
1521 if(g_rds_thread_running)
1522 g_rds_thread_terminate = 1;
1523
1524 close(g_radio_fd);
1525 printf("Terminating..\n\n");
1526 exit(1);
1527 }
main()1528 int main()
1529 {
1530 char choice[100];
1531 char exit_flag;
1532 int ret;
1533 struct sigaction sa;
1534
1535 printf("** TI Kernel Space FM Driver Test Application **\n");
1536
1537 printf("Opening device '%s'\n",DEFAULT_RADIO_DEVICE);
1538 g_radio_fd = open(DEFAULT_RADIO_DEVICE, O_RDWR);
1539 if(g_radio_fd < 0)
1540 {
1541 printf("Unable to open %s \nTerminating..\n",DEFAULT_RADIO_DEVICE);
1542 return 0;
1543 }
1544 ret = fmapp_read_anddisplay_capabilities();
1545 if(ret< 0)
1546 {
1547 close(g_radio_fd);
1548 return ret;
1549 }
1550 /* to handle ctrl + c and kill signals */
1551 memset(&sa, 0, sizeof(sa));
1552 sa.sa_handler = sig_handler;
1553 sigaction(SIGTERM, &sa, NULL);
1554 sigaction(SIGINT, &sa, NULL);
1555
1556 exit_flag = 1;
1557 while(exit_flag)
1558 {
1559 printf("1 FM RX\n");
1560 printf("2 FM TX\n");
1561 printf("3 Exit\n");
1562 fgets(choice, sizeof(choice), stdin);
1563
1564 switch(atoi(choice))
1565 {
1566 case 1: /* FM RX */
1567 fmapp_execute_rx_command();
1568 break;
1569 case 2: /* FM TX */
1570 fmapp_execute_tx_command();
1571 break;
1572 case 3:
1573 printf("Terminating..\n\n");
1574 exit_flag = 0;
1575 break;
1576 default:
1577 printf("Invalid choice , try again\n");
1578 continue;
1579 }
1580 }
1581 if(g_rds_thread_running)
1582 g_rds_thread_terminate = 1; // Terminate RDS thread
1583
1584 close(g_radio_fd);
1585 return 0;
1586 }
1587
1588
1589