1
0
mirror of https://git.FreeBSD.org/src.git synced 2024-10-19 02:29:40 +00:00

Fix off-by-one errors and potential buffer overruns

WRT handling file and link names that reach the allowed
maximum for old tar and ustar archive formats.

PR:		bin/40466
Submitted by:	Cyrille Lefevre <email in the PR> (portions)
Reviewed by:	freebsd-arch (silence)
MFC after:	1 month
This commit is contained in:
Yaroslav Tykhiy 2004-11-13 10:56:35 +00:00
parent 53d0031d37
commit 5512dc5460
Notes: svn2git 2020-12-20 02:59:44 +00:00
svn path=/head/; revision=137645

View File

@ -387,7 +387,13 @@ tar_rd(ARCHD *arcn, char *buf)
* copy out the name and values in the stat buffer * copy out the name and values in the stat buffer
*/ */
hd = (HD_TAR *)buf; hd = (HD_TAR *)buf;
arcn->nlen = l_strncpy(arcn->name, hd->name, sizeof(arcn->name) - 1); /*
* old tar format specifies the name always be null-terminated,
* but let's be robust to broken archives.
* the same applies to handling links below.
*/
arcn->nlen = l_strncpy(arcn->name, hd->name,
MIN(sizeof(hd->name), sizeof(arcn->name)) - 1);
arcn->name[arcn->nlen] = '\0'; arcn->name[arcn->nlen] = '\0';
arcn->sb.st_mode = (mode_t)(asc_ul(hd->mode,sizeof(hd->mode),OCT) & arcn->sb.st_mode = (mode_t)(asc_ul(hd->mode,sizeof(hd->mode),OCT) &
0xfff); 0xfff);
@ -417,7 +423,7 @@ tar_rd(ARCHD *arcn, char *buf)
*/ */
arcn->type = PAX_SLK; arcn->type = PAX_SLK;
arcn->ln_nlen = l_strncpy(arcn->ln_name, hd->linkname, arcn->ln_nlen = l_strncpy(arcn->ln_name, hd->linkname,
sizeof(arcn->ln_name) - 1); MIN(sizeof(hd->linkname), sizeof(arcn->ln_name)) - 1);
arcn->ln_name[arcn->ln_nlen] = '\0'; arcn->ln_name[arcn->ln_nlen] = '\0';
arcn->sb.st_mode |= S_IFLNK; arcn->sb.st_mode |= S_IFLNK;
break; break;
@ -429,7 +435,7 @@ tar_rd(ARCHD *arcn, char *buf)
arcn->type = PAX_HLK; arcn->type = PAX_HLK;
arcn->sb.st_nlink = 2; arcn->sb.st_nlink = 2;
arcn->ln_nlen = l_strncpy(arcn->ln_name, hd->linkname, arcn->ln_nlen = l_strncpy(arcn->ln_name, hd->linkname,
sizeof(arcn->ln_name) - 1); MIN(sizeof(hd->linkname), sizeof(arcn->ln_name)) - 1);
arcn->ln_name[arcn->ln_nlen] = '\0'; arcn->ln_name[arcn->ln_nlen] = '\0';
/* /*
@ -533,7 +539,7 @@ tar_wr(ARCHD *arcn)
case PAX_SLK: case PAX_SLK:
case PAX_HLK: case PAX_HLK:
case PAX_HRG: case PAX_HRG:
if (arcn->ln_nlen > (int)sizeof(hd->linkname)) { if (arcn->ln_nlen >= (int)sizeof(hd->linkname)) {
paxwarn(1,"Link name too long for tar %s", arcn->ln_name); paxwarn(1,"Link name too long for tar %s", arcn->ln_name);
return(1); return(1);
} }
@ -749,12 +755,19 @@ ustar_rd(ARCHD *arcn, char *buf)
*/ */
dest = arcn->name; dest = arcn->name;
if (*(hd->prefix) != '\0') { if (*(hd->prefix) != '\0') {
cnt = l_strncpy(dest, hd->prefix, sizeof(arcn->name) - 2); cnt = l_strncpy(dest, hd->prefix,
MIN(sizeof(hd->prefix), sizeof(arcn->name) - 2));
dest += cnt; dest += cnt;
*dest++ = '/'; *dest++ = '/';
cnt++; cnt++;
} }
arcn->nlen = cnt + l_strncpy(dest, hd->name, sizeof(arcn->name) - cnt); /*
* ustar format specifies the name may be unterminated
* if it fills the entire field. this also applies to
* the prefix and the linkname.
*/
arcn->nlen = cnt + l_strncpy(dest, hd->name,
MIN(sizeof(hd->name), sizeof(arcn->name) - cnt - 1));
arcn->name[arcn->nlen] = '\0'; arcn->name[arcn->nlen] = '\0';
/* /*
@ -848,7 +861,7 @@ ustar_rd(ARCHD *arcn, char *buf)
* copy the link name * copy the link name
*/ */
arcn->ln_nlen = l_strncpy(arcn->ln_name, hd->linkname, arcn->ln_nlen = l_strncpy(arcn->ln_name, hd->linkname,
sizeof(arcn->ln_name) - 1); MIN(sizeof(hd->linkname), sizeof(arcn->ln_name) - 1));
arcn->ln_name[arcn->ln_nlen] = '\0'; arcn->ln_name[arcn->ln_nlen] = '\0';
break; break;
case CONTTYPE: case CONTTYPE:
@ -900,7 +913,7 @@ ustar_wr(ARCHD *arcn)
*/ */
if (((arcn->type == PAX_SLK) || (arcn->type == PAX_HLK) || if (((arcn->type == PAX_SLK) || (arcn->type == PAX_HLK) ||
(arcn->type == PAX_HRG)) && (arcn->type == PAX_HRG)) &&
(arcn->ln_nlen >= (int)sizeof(hd->linkname))) { (arcn->ln_nlen > (int)sizeof(hd->linkname))) {
paxwarn(1, "Link name too long for ustar %s", arcn->ln_name); paxwarn(1, "Link name too long for ustar %s", arcn->ln_name);
return(1); return(1);
} }
@ -925,17 +938,16 @@ ustar_wr(ARCHD *arcn)
* occur, we remove the / and copy the first part to the prefix * occur, we remove the / and copy the first part to the prefix
*/ */
*pt = '\0'; *pt = '\0';
l_strncpy(hd->prefix, arcn->name, sizeof(hd->prefix) - 1); l_strncpy(hd->prefix, arcn->name, sizeof(hd->prefix));
*pt++ = '/'; *pt++ = '/';
} else } else
memset(hd->prefix, 0, sizeof(hd->prefix)); memset(hd->prefix, 0, sizeof(hd->prefix));
/* /*
* copy the name part. this may be the whole path or the part after * copy the name part. this may be the whole path or the part after
* the prefix * the prefix. both the name and prefix may fill the entire field.
*/ */
l_strncpy(hd->name, pt, sizeof(hd->name) - 1); l_strncpy(hd->name, pt, sizeof(hd->name));
hd->name[sizeof(hd->name) - 1] = '\0';
/* /*
* set the fields in the header that are type dependent * set the fields in the header that are type dependent
@ -978,8 +990,8 @@ ustar_wr(ARCHD *arcn)
hd->typeflag = SYMTYPE; hd->typeflag = SYMTYPE;
else else
hd->typeflag = LNKTYPE; hd->typeflag = LNKTYPE;
l_strncpy(hd->linkname,arcn->ln_name, sizeof(hd->linkname) - 1); /* the link name may occupy the entire field in ustar */
hd->linkname[sizeof(hd->linkname) - 1] = '\0'; l_strncpy(hd->linkname,arcn->ln_name, sizeof(hd->linkname));
memset(hd->devmajor, 0, sizeof(hd->devmajor)); memset(hd->devmajor, 0, sizeof(hd->devmajor));
memset(hd->devminor, 0, sizeof(hd->devminor)); memset(hd->devminor, 0, sizeof(hd->devminor));
if (ul_oct((u_long)0L, hd->size, sizeof(hd->size), 3)) if (ul_oct((u_long)0L, hd->size, sizeof(hd->size), 3))
@ -1072,9 +1084,9 @@ name_split(char *name, int len)
* check to see if the file name is small enough to fit in the name * check to see if the file name is small enough to fit in the name
* field. if so just return a pointer to the name. * field. if so just return a pointer to the name.
*/ */
if (len < TNMSZ) if (len <= TNMSZ)
return(name); return(name);
if (len > (TPFSZ + TNMSZ)) if (len > (TPFSZ + TNMSZ + 1))
return(NULL); return(NULL);
/* /*
@ -1083,7 +1095,7 @@ name_split(char *name, int len)
* to find the biggest piece to fit in the name field (or the smallest * to find the biggest piece to fit in the name field (or the smallest
* prefix we can find) * prefix we can find)
*/ */
start = name + len - TNMSZ; start = name + len - TNMSZ - 1;
while ((*start != '\0') && (*start != '/')) while ((*start != '\0') && (*start != '/'))
++start; ++start;
@ -1101,7 +1113,7 @@ name_split(char *name, int len)
* the file would then expand on extract to //str. The len == 0 below * the file would then expand on extract to //str. The len == 0 below
* makes this special case follow the spec to the letter. * makes this special case follow the spec to the letter.
*/ */
if ((len >= TPFSZ) || (len == 0)) if ((len > TPFSZ) || (len == 0))
return(NULL); return(NULL);
/* /*