• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2011-2012 - Mauro Carvalho Chehab
3  * Copyright (c) 2012-2014 - Andre Roth <neolynx@gmail.com>
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU Lesser General Public License as published by
7  * the Free Software Foundation version 2.1 of the License.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17  * Or, point your browser to http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
18  *
19  */
20 
21 #include <libdvbv5/nit.h>
22 #include <libdvbv5/dvb-fe.h>
23 
24 #if __GNUC__ >= 9
25 #pragma GCC diagnostic ignored "-Waddress-of-packed-member"
26 #endif
27 
dvb_table_nit_init(struct dvb_v5_fe_parms * parms,const uint8_t * buf,ssize_t buflen,struct dvb_table_nit ** table)28 ssize_t dvb_table_nit_init(struct dvb_v5_fe_parms *parms, const uint8_t *buf,
29 			ssize_t buflen, struct dvb_table_nit **table)
30 {
31 	const uint8_t *p = buf, *endbuf = buf + buflen;
32 	struct dvb_table_nit *nit;
33 	struct dvb_table_nit_transport **head;
34 	struct dvb_desc **head_desc;
35 	size_t size;
36 
37 	size = offsetof(struct dvb_table_nit, descriptor);
38 	if (p + size > endbuf) {
39 		dvb_logerr("%s: short read %zd/%zd bytes", __func__,
40 			   endbuf - p, size);
41 		return -1;
42 	}
43 
44 	if (buf[0] != DVB_TABLE_NIT && (buf[0] != DVB_TABLE_NIT2)) {
45 		dvb_logerr("%s: invalid marker 0x%02x, should be 0x%02x or 0x%02x",
46 				__func__, buf[0], DVB_TABLE_NIT, DVB_TABLE_NIT2);
47 		return -2;
48 	}
49 
50 	if (!*table) {
51 		*table = calloc(sizeof(struct dvb_table_nit), 1);
52 		if (!*table) {
53 			dvb_logerr("%s: out of memory", __func__);
54 			return -3;
55 		}
56 	}
57 	nit = *table;
58 	memcpy(nit, p, size);
59 	p += size;
60 	dvb_table_header_init(&nit->header);
61 
62 	bswap16(nit->bitfield);
63 
64 	/* find end of current lists */
65 	head_desc = &nit->descriptor;
66 	while (*head_desc != NULL)
67 		head_desc = &(*head_desc)->next;
68 	head = &nit->transport;
69 	while (*head != NULL)
70 		head = &(*head)->next;
71 
72 	size = nit->desc_length;
73 	if (p + size > endbuf) {
74 		dvb_logerr("%s: short read %zd/%zd bytes", __func__,
75 			   endbuf - p, size);
76 		return -4;
77 	}
78 	if (dvb_desc_parse(parms, p, size, head_desc) != 0) {
79 		return -5;
80 	}
81 	p += size;
82 
83 	size = sizeof(union dvb_table_nit_transport_header);
84 	if (p + size > endbuf) {
85 		dvb_logerr("%s: short read %zd/%zd bytes", __func__,
86 			   endbuf - p, size);
87 		return -6;
88 	}
89 	p += size;
90 
91 	size = offsetof(struct dvb_table_nit_transport, descriptor);
92 	while (p + size <= endbuf) {
93 		struct dvb_table_nit_transport *transport;
94 
95 		transport = malloc(sizeof(struct dvb_table_nit_transport));
96 		if (!transport) {
97 			dvb_logerr("%s: out of memory", __func__);
98 			return -7;
99 		}
100 		memcpy(transport, p, size);
101 		p += size;
102 
103 		bswap16(transport->transport_id);
104 		bswap16(transport->network_id);
105 		bswap16(transport->bitfield);
106 		transport->descriptor = NULL;
107 		transport->next = NULL;
108 
109 		*head = transport;
110 		head = &(*head)->next;
111 
112 		/* parse the descriptors */
113 		if (transport->desc_length > 0) {
114 			uint16_t desc_length = transport->desc_length;
115 			if (p + desc_length > endbuf) {
116 				dvb_logwarn("%s: descriptors short read %zd/%d bytes", __func__,
117 					   endbuf - p, desc_length);
118 				desc_length = endbuf - p;
119 			}
120 			if (dvb_desc_parse(parms, p, desc_length,
121 					      &transport->descriptor) != 0) {
122 				return -8;
123 			}
124 			p += desc_length;
125 		}
126 	}
127 	if (endbuf - p)
128 		dvb_logwarn("%s: %zu spurious bytes at the end",
129 			   __func__, endbuf - p);
130 	return p - buf;
131 }
132 
dvb_table_nit_free(struct dvb_table_nit * nit)133 void dvb_table_nit_free(struct dvb_table_nit *nit)
134 {
135 	struct dvb_table_nit_transport *transport = nit->transport;
136 	dvb_desc_free(&nit->descriptor);
137 	while (transport) {
138 		dvb_desc_free(&transport->descriptor);
139 		struct dvb_table_nit_transport *tmp = transport;
140 		transport = transport->next;
141 		free(tmp);
142 	}
143 	free(nit);
144 }
145 
dvb_table_nit_print(struct dvb_v5_fe_parms * parms,struct dvb_table_nit * nit)146 void dvb_table_nit_print(struct dvb_v5_fe_parms *parms, struct dvb_table_nit *nit)
147 {
148 	const struct dvb_table_nit_transport *transport = nit->transport;
149 	uint16_t transports = 0;
150 
151 	dvb_loginfo("NIT");
152 	dvb_table_header_print(parms, &nit->header);
153 	dvb_loginfo("| desc_length   %d", nit->desc_length);
154 	dvb_desc_print(parms, nit->descriptor);
155 	while (transport) {
156 		dvb_loginfo("|- transport %04x network %04x", transport->transport_id, transport->network_id);
157 		dvb_desc_print(parms, transport->descriptor);
158 		transport = transport->next;
159 		transports++;
160 	}
161 	dvb_loginfo("|_  %d transports", transports);
162 }
163 
dvb_table_nit_descriptor_handler(struct dvb_v5_fe_parms * parms,struct dvb_table_nit * nit,enum descriptors descriptor,nit_handler_callback_t * call_nit,nit_tran_handler_callback_t * call_tran,void * priv)164 void dvb_table_nit_descriptor_handler(
165 			    struct dvb_v5_fe_parms *parms,
166 			    struct dvb_table_nit *nit,
167 			    enum descriptors descriptor,
168 			    nit_handler_callback_t *call_nit,
169 			    nit_tran_handler_callback_t *call_tran,
170 			    void *priv)
171 {
172 	if (call_nit || parms->verbose) {
173 		dvb_desc_find(struct dvb_desc, desc, nit,
174 			      descriptor) {
175 			if (call_nit)
176 				call_nit(nit, desc, priv);
177 			else
178 				dvb_logwarn("descriptor %s found on NIT but unhandled",
179 					   dvb_descriptors[descriptor].name);
180 
181 		}
182 	}
183 	if (!call_tran && !parms->verbose)
184 		return;
185 
186 	dvb_nit_transport_foreach(tran, nit) {
187 		dvb_desc_find(struct dvb_desc, desc, tran,
188 			      descriptor) {
189 			if (call_tran)
190 				call_tran(nit, tran, desc, priv);
191 			else
192 				dvb_logwarn("descriptor %s found on NIT transport but unhandled",
193 					   dvb_descriptors[descriptor].name);
194 		}
195 	}
196 }
197