Skip to content

Commit 4a1aa2b

Browse files
committed
Issue 296 - Avoid potential buffer overflow.
1 parent bfc708d commit 4a1aa2b

File tree

2 files changed

+50
-6
lines changed

2 files changed

+50
-6
lines changed

src/encode_aprs.c

+9-2
Original file line numberDiff line numberDiff line change
@@ -81,16 +81,23 @@ typedef struct position_s {
8181

8282
static int set_norm_position (char symtab, char symbol, double dlat, double dlong, int ambiguity, position_t *presult)
8383
{
84+
// An over zealous compiler might complain about l*itude_to_str writing
85+
// N characters plus nul to an N character field so we stick it into a
86+
// larger temp then copy the desired number of bytes. (Issue 296)
8487

85-
latitude_to_str (dlat, ambiguity, presult->lat);
88+
char stemp[16];
89+
90+
latitude_to_str (dlat, ambiguity, stemp);
91+
memcpy (presult->lat, stemp, sizeof(presult->lat));
8692

8793
if (symtab != '/' && symtab != '\\' && ! isdigit(symtab) && ! isupper(symtab)) {
8894
text_color_set(DW_COLOR_ERROR);
8995
dw_printf ("Symbol table identifier is not one of / \\ 0-9 A-Z\n");
9096
}
9197
presult->sym_table_id = symtab;
9298

93-
longitude_to_str (dlong, ambiguity, presult->lon);
99+
longitude_to_str (dlong, ambiguity, stemp);
100+
memcpy (presult->lon, stemp, sizeof(presult->lon));
94101

95102
if (symbol < '!' || symbol > '~') {
96103
text_color_set(DW_COLOR_ERROR);

src/latlong.c

+41-4
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,11 @@
5555
* ambiguity - If 1, 2, 3, or 4, blank out that many trailing digits.
5656
*
5757
* Outputs: slat - String in format ddmm.mm[NS]
58-
* Should always be exactly 8 characters + NUL.
58+
* Must always be exactly 8 characters + NUL.
59+
* Put in leading zeros if necessary.
60+
* We must have exactly ddmm.mm and hemisphere because
61+
* the APRS position report has fixed width fields.
62+
* Trailing digits can be blanked for position ambiguity.
5963
*
6064
* Returns: None
6165
*
@@ -101,13 +105,22 @@ void latitude_to_str (double dlat, int ambiguity, char *slat)
101105
ideg = (int)dlat;
102106
dmin = (dlat - ideg) * 60.;
103107

108+
// dmin is known to be in range of 0 <= dmin < 60.
109+
110+
// Minutes must be exactly like 99.99 with leading zeros,
111+
// if needed, to make it fixed width.
112+
// Two digits, decimal point, two digits, nul terminator.
113+
104114
snprintf (smin, sizeof(smin), "%05.2f", dmin);
105115
/* Due to roundoff, 59.9999 could come out as "60.00" */
106116
if (smin[0] == '6') {
107117
smin[0] = '0';
108118
ideg++;
109119
}
110120

121+
// Assumes slat can hold 8 characters + nul.
122+
// Degrees must be exactly 2 digits, with leading zero, if needed.
123+
111124
sprintf (slat, "%02d%s%c", ideg, smin, hemi);
112125

113126
if (ambiguity >= 1) {
@@ -135,9 +148,12 @@ void latitude_to_str (double dlat, int ambiguity, char *slat)
135148
* Inputs: dlong - Floating point degrees.
136149
* ambiguity - If 1, 2, 3, or 4, blank out that many trailing digits.
137150
*
138-
* Outputs: slat - String in format dddmm.mm[NS]
139-
* Should always be exactly 9 characters + NUL.
140-
*
151+
* Outputs: slong - String in format dddmm.mm[NS]
152+
* Must always be exactly 9 characters + NUL.
153+
* Put in leading zeros if necessary.
154+
* We must have exactly dddmm.mm and hemisphere because
155+
* the APRS position report has fixed width fields.
156+
* Trailing digits can be blanked for position ambiguity.
141157
* Returns: None
142158
*
143159
*----------------------------------------------------------------*/
@@ -178,7 +194,11 @@ void longitude_to_str (double dlong, int ambiguity, char *slong)
178194
ideg++;
179195
}
180196

197+
// Assumes slong can hold 9 characters + nul.
198+
// Degrees must be exactly 3 digits, with leading zero, if needed.
199+
181200
sprintf (slong, "%03d%s%c", ideg, smin, hemi);
201+
182202
/*
183203
* The spec says position ambiguity in latitude also
184204
* applies to longitude automatically.
@@ -903,6 +923,14 @@ int main (int argc, char *argv[])
903923
latitude_to_str (45.999830, 4, result);
904924
if (strcmp(result, "45 . N") != 0) { errors++; dw_printf ("Error 1.8: Did not expect \"%s\"\n", result); }
905925

926+
// Test for leading zeros for small values. Result must be fixed width.
927+
928+
latitude_to_str (0.016666666, 0, result);
929+
if (strcmp(result, "0001.00N") != 0) { errors++; dw_printf ("Error 1.9: Did not expect \"%s\"\n", result); }
930+
931+
latitude_to_str (-1.999999, 0, result);
932+
if (strcmp(result, "0200.00S") != 0) { errors++; dw_printf ("Error 1.10: Did not expect \"%s\"\n", result); }
933+
906934
/* Longitude to APRS format. */
907935

908936
longitude_to_str (45.25, 0, result);
@@ -931,6 +959,15 @@ int main (int argc, char *argv[])
931959
longitude_to_str (45.999830, 4, result);
932960
if (strcmp(result, "045 . E") != 0) { errors++; dw_printf ("Error 2.8: Did not expect \"%s\"\n", result); }
933961

962+
// Test for leading zeros for small values. Result must be fixed width.
963+
964+
longitude_to_str (0.016666666, 0, result);
965+
if (strcmp(result, "00001.00E") != 0) { errors++; dw_printf ("Error 2.9: Did not expect \"%s\"\n", result); }
966+
967+
longitude_to_str (-1.999999, 0, result);
968+
if (strcmp(result, "00200.00W") != 0) { errors++; dw_printf ("Error 2.10: Did not expect \"%s\"\n", result); }
969+
970+
934971
/* Compressed format. */
935972
/* Protocol spec example has <*e7 but I got <*e8 due to rounding rather than truncation to integer. */
936973

0 commit comments

Comments
 (0)