diff options
Diffstat (limited to 'fs/binfmt_elf_fdpic.c')
| -rw-r--r-- | fs/binfmt_elf_fdpic.c | 230 | 
1 files changed, 79 insertions, 151 deletions
diff --git a/fs/binfmt_elf_fdpic.c b/fs/binfmt_elf_fdpic.c index 63039ed9576..fe2a643ee00 100644 --- a/fs/binfmt_elf_fdpic.c +++ b/fs/binfmt_elf_fdpic.c @@ -56,7 +56,7 @@ typedef char *elf_caddr_t;  MODULE_LICENSE("GPL"); -static int load_elf_fdpic_binary(struct linux_binprm *, struct pt_regs *); +static int load_elf_fdpic_binary(struct linux_binprm *);  static int elf_fdpic_fetch_phdrs(struct elf_fdpic_params *, struct file *);  static int elf_fdpic_map_file(struct elf_fdpic_params *, struct file *,  			      struct mm_struct *, const char *); @@ -91,7 +91,8 @@ static struct linux_binfmt elf_fdpic_format = {  static int __init init_elf_fdpic_binfmt(void)  { -	return register_binfmt(&elf_fdpic_format); +	register_binfmt(&elf_fdpic_format); +	return 0;  }  static void __exit exit_elf_fdpic_binfmt(void) @@ -110,7 +111,7 @@ static int is_elf_fdpic(struct elfhdr *hdr, struct file *file)  		return 0;  	if (!elf_check_arch(hdr) || !elf_check_fdpic(hdr))  		return 0; -	if (!file->f_op || !file->f_op->mmap) +	if (!file->f_op->mmap)  		return 0;  	return 1;  } @@ -163,10 +164,10 @@ static int elf_fdpic_fetch_phdrs(struct elf_fdpic_params *params,  /*   * load an fdpic binary into various bits of memory   */ -static int load_elf_fdpic_binary(struct linux_binprm *bprm, -				 struct pt_regs *regs) +static int load_elf_fdpic_binary(struct linux_binprm *bprm)  {  	struct elf_fdpic_params exec_params, interp_params; +	struct pt_regs *regs = current_pt_regs();  	struct elf_phdr *phdr;  	unsigned long stack_size, entryaddr;  #ifdef ELF_FDPIC_PLAT_INIT @@ -245,8 +246,7 @@ static int load_elf_fdpic_binary(struct linux_binprm *bprm,  			 * mm->dumpable = 0 regardless of the interpreter's  			 * permissions.  			 */ -			if (file_permission(interpreter, MAY_READ) < 0) -				bprm->interp_flags |= BINPRM_FLAGS_ENFORCE_NONDUMP; +			would_dump(bprm, interpreter);  			retval = kernel_read(interpreter, 0, bprm->buf,  					     BINPRM_BUF_SIZE); @@ -335,8 +335,6 @@ static int load_elf_fdpic_binary(struct linux_binprm *bprm,  	current->mm->context.exec_fdpic_loadmap = 0;  	current->mm->context.interp_fdpic_loadmap = 0; -	current->flags &= ~PF_FORKNOEXEC; -  #ifdef CONFIG_MMU  	elf_fdpic_arch_lay_out_mm(&exec_params,  				  &interp_params, @@ -391,21 +389,17 @@ static int load_elf_fdpic_binary(struct linux_binprm *bprm,  	    (executable_stack == EXSTACK_DEFAULT && VM_STACK_FLAGS & VM_EXEC))  		stack_prot |= PROT_EXEC; -	down_write(¤t->mm->mmap_sem); -	current->mm->start_brk = do_mmap(NULL, 0, stack_size, stack_prot, +	current->mm->start_brk = vm_mmap(NULL, 0, stack_size, stack_prot,  					 MAP_PRIVATE | MAP_ANONYMOUS |  					 MAP_UNINITIALIZED | MAP_GROWSDOWN,  					 0);  	if (IS_ERR_VALUE(current->mm->start_brk)) { -		up_write(¤t->mm->mmap_sem);  		retval = current->mm->start_brk;  		current->mm->start_brk = 0;  		goto error_kill;  	} -	up_write(¤t->mm->mmap_sem); -  	current->mm->brk = current->mm->start_brk;  	current->mm->context.end_brk = current->mm->start_brk;  	current->mm->context.end_brk += @@ -414,7 +408,6 @@ static int load_elf_fdpic_binary(struct linux_binprm *bprm,  #endif  	install_exec_creds(bprm); -	current->flags &= ~PF_FORKNOEXEC;  	if (create_elf_fdpic_tables(bprm, current->mm,  				    &exec_params, &interp_params) < 0)  		goto error_kill; @@ -490,7 +483,6 @@ static int create_elf_fdpic_tables(struct linux_binprm *bprm,  	size_t platform_len = 0, len;  	char *k_platform, *k_base_platform;  	char __user *u_platform, *u_base_platform, *p; -	long hwcap;  	int loop;  	int nr;	/* reset for each csp adjustment */ @@ -509,8 +501,6 @@ static int create_elf_fdpic_tables(struct linux_binprm *bprm,  		return -EFAULT;  #endif -	hwcap = ELF_HWCAP; -  	/*  	 * If this architecture has a platform capability string, copy it  	 * to userspace.  In some cases (Sparc), this info is impossible @@ -624,7 +614,10 @@ static int create_elf_fdpic_tables(struct linux_binprm *bprm,  	nr = 0;  	csp -= DLINFO_ITEMS * 2 * sizeof(unsigned long); -	NEW_AUX_ENT(AT_HWCAP,	hwcap); +	NEW_AUX_ENT(AT_HWCAP,	ELF_HWCAP); +#ifdef ELF_HWCAP2 +	NEW_AUX_ENT(AT_HWCAP2,	ELF_HWCAP2); +#endif  	NEW_AUX_ENT(AT_PAGESZ,	PAGE_SIZE);  	NEW_AUX_ENT(AT_CLKTCK,	CLOCKS_PER_SEC);  	NEW_AUX_ENT(AT_PHDR,	exec_params->ph_addr); @@ -633,10 +626,10 @@ static int create_elf_fdpic_tables(struct linux_binprm *bprm,  	NEW_AUX_ENT(AT_BASE,	interp_params->elfhdr_addr);  	NEW_AUX_ENT(AT_FLAGS,	0);  	NEW_AUX_ENT(AT_ENTRY,	exec_params->entry_addr); -	NEW_AUX_ENT(AT_UID,	(elf_addr_t) cred->uid); -	NEW_AUX_ENT(AT_EUID,	(elf_addr_t) cred->euid); -	NEW_AUX_ENT(AT_GID,	(elf_addr_t) cred->gid); -	NEW_AUX_ENT(AT_EGID,	(elf_addr_t) cred->egid); +	NEW_AUX_ENT(AT_UID,	(elf_addr_t) from_kuid_munged(cred->user_ns, cred->uid)); +	NEW_AUX_ENT(AT_EUID,	(elf_addr_t) from_kuid_munged(cred->user_ns, cred->euid)); +	NEW_AUX_ENT(AT_GID,	(elf_addr_t) from_kgid_munged(cred->user_ns, cred->gid)); +	NEW_AUX_ENT(AT_EGID,	(elf_addr_t) from_kgid_munged(cred->user_ns, cred->egid));  	NEW_AUX_ENT(AT_SECURE,	security_bprm_secureexec(bprm));  	NEW_AUX_ENT(AT_EXECFN,	bprm->exec); @@ -916,7 +909,7 @@ static int elf_fdpic_map_file(struct elf_fdpic_params *params,  dynamic_error:  	printk("ELF FDPIC %s with invalid DYNAMIC section (inode=%lu)\n", -	       what, file->f_path.dentry->d_inode->i_ino); +	       what, file_inode(file)->i_ino);  	return -ELIBBAD;  } @@ -933,7 +926,6 @@ static int elf_fdpic_map_file_constdisp_on_uclinux(  	struct elf32_fdpic_loadseg *seg;  	struct elf32_phdr *phdr;  	unsigned long load_addr, base = ULONG_MAX, top = 0, maddr = 0, mflags; -	loff_t fpos;  	int loop, ret;  	load_addr = params->load_addr; @@ -957,10 +949,8 @@ static int elf_fdpic_map_file_constdisp_on_uclinux(  	if (params->flags & ELF_FDPIC_FLAG_EXECUTABLE)  		mflags |= MAP_EXECUTABLE; -	down_write(&mm->mmap_sem); -	maddr = do_mmap(NULL, load_addr, top - base, +	maddr = vm_mmap(NULL, load_addr, top - base,  			PROT_READ | PROT_WRITE | PROT_EXEC, mflags, 0); -	up_write(&mm->mmap_sem);  	if (IS_ERR_VALUE(maddr))  		return (int) maddr; @@ -973,14 +963,12 @@ static int elf_fdpic_map_file_constdisp_on_uclinux(  		if (params->phdrs[loop].p_type != PT_LOAD)  			continue; -		fpos = phdr->p_offset; -  		seg->addr = maddr + (phdr->p_vaddr - base);  		seg->p_vaddr = phdr->p_vaddr;  		seg->p_memsz = phdr->p_memsz; -		ret = file->f_op->read(file, (void *) seg->addr, -				       phdr->p_filesz, &fpos); +		ret = read_code(file, seg->addr, phdr->p_offset, +				       phdr->p_filesz);  		if (ret < 0)  			return ret; @@ -1098,10 +1086,8 @@ static int elf_fdpic_map_file_by_direct_mmap(struct elf_fdpic_params *params,  		/* create the mapping */  		disp = phdr->p_vaddr & ~PAGE_MASK; -		down_write(&mm->mmap_sem); -		maddr = do_mmap(file, maddr, phdr->p_memsz + disp, prot, flags, +		maddr = vm_mmap(file, maddr, phdr->p_memsz + disp, prot, flags,  				phdr->p_offset - disp); -		up_write(&mm->mmap_sem);  		kdebug("mmap[%d] <file> sz=%lx pr=%x fl=%x of=%lx --> %08lx",  		       loop, phdr->p_memsz + disp, prot, flags, @@ -1145,10 +1131,8 @@ static int elf_fdpic_map_file_by_direct_mmap(struct elf_fdpic_params *params,  			unsigned long xmaddr;  			flags |= MAP_FIXED | MAP_ANONYMOUS; -			down_write(&mm->mmap_sem); -			xmaddr = do_mmap(NULL, xaddr, excess - excess1, +			xmaddr = vm_mmap(NULL, xaddr, excess - excess1,  					 prot, flags, 0); -			up_write(&mm->mmap_sem);  			kdebug("mmap[%d] <anon>"  			       " ad=%lx sz=%lx pr=%x fl=%x of=0 --> %08lx", @@ -1217,7 +1201,7 @@ static int maydump(struct vm_area_struct *vma, unsigned long mm_flags)  	int dump_ok;  	/* Do not dump I/O mapped devices or special mappings */ -	if (vma->vm_flags & (VM_IO | VM_RESERVED)) { +	if (vma->vm_flags & VM_IO) {  		kdcore("%08lx: %08lx: no (IO)", vma->vm_start, vma->vm_flags);  		return 0;  	} @@ -1232,7 +1216,7 @@ static int maydump(struct vm_area_struct *vma, unsigned long mm_flags)  	/* By default, dump shared memory if mapped from an anonymous file. */  	if (vma->vm_flags & VM_SHARED) { -		if (vma->vm_file->f_path.dentry->d_inode->i_nlink == 0) { +		if (file_inode(vma->vm_file)->i_nlink == 0) {  			dump_ok = test_bit(MMF_DUMP_ANON_SHARED, &mm_flags);  			kdcore("%08lx: %08lx: %s (share)", vma->vm_start,  			       vma->vm_flags, dump_ok ? "yes" : "no"); @@ -1283,35 +1267,17 @@ static int notesize(struct memelfnote *en)  /* #define DEBUG */ -#define DUMP_WRITE(addr, nr, foffset)	\ -	do { if (!dump_write(file, (addr), (nr))) return 0; *foffset += (nr); } while(0) - -static int alignfile(struct file *file, loff_t *foffset) -{ -	static const char buf[4] = { 0, }; -	DUMP_WRITE(buf, roundup(*foffset, 4) - *foffset, foffset); -	return 1; -} - -static int writenote(struct memelfnote *men, struct file *file, -			loff_t *foffset) +static int writenote(struct memelfnote *men, struct coredump_params *cprm)  {  	struct elf_note en;  	en.n_namesz = strlen(men->name) + 1;  	en.n_descsz = men->datasz;  	en.n_type = men->type; -	DUMP_WRITE(&en, sizeof(en), foffset); -	DUMP_WRITE(men->name, en.n_namesz, foffset); -	if (!alignfile(file, foffset)) -		return 0; -	DUMP_WRITE(men->data, men->datasz, foffset); -	if (!alignfile(file, foffset)) -		return 0; - -	return 1; +	return dump_emit(cprm, &en, sizeof(en)) && +		dump_emit(cprm, men->name, en.n_namesz) && dump_align(cprm, 4) && +		dump_emit(cprm, men->data, men->datasz) && dump_align(cprm, 4);  } -#undef DUMP_WRITE  static inline void fill_elf_fdpic_header(struct elfhdr *elf, int segs)  { @@ -1388,8 +1354,11 @@ static void fill_prstatus(struct elf_prstatus *prstatus,  		cputime_to_timeval(cputime.utime, &prstatus->pr_utime);  		cputime_to_timeval(cputime.stime, &prstatus->pr_stime);  	} else { -		cputime_to_timeval(p->utime, &prstatus->pr_utime); -		cputime_to_timeval(p->stime, &prstatus->pr_stime); +		cputime_t utime, stime; + +		task_cputime(p, &utime, &stime); +		cputime_to_timeval(utime, &prstatus->pr_utime); +		cputime_to_timeval(stime, &prstatus->pr_stime);  	}  	cputime_to_timeval(p->signal->cutime, &prstatus->pr_cutime);  	cputime_to_timeval(p->signal->cstime, &prstatus->pr_cstime); @@ -1433,8 +1402,8 @@ static int fill_psinfo(struct elf_prpsinfo *psinfo, struct task_struct *p,  	psinfo->pr_flag = p->flags;  	rcu_read_lock();  	cred = __task_cred(p); -	SET_UID(psinfo->pr_uid, cred->uid); -	SET_GID(psinfo->pr_gid, cred->gid); +	SET_UID(psinfo->pr_uid, from_kuid_munged(cred->user_ns, cred->uid)); +	SET_GID(psinfo->pr_gid, from_kgid_munged(cred->user_ns, cred->gid));  	rcu_read_unlock();  	strncpy(psinfo->pr_fname, p->comm, sizeof(psinfo->pr_fname)); @@ -1513,66 +1482,40 @@ static void fill_extnum_info(struct elfhdr *elf, struct elf_shdr *shdr4extnum,  /*   * dump the segments for an MMU process   */ -#ifdef CONFIG_MMU -static int elf_fdpic_dump_segments(struct file *file, size_t *size, -			   unsigned long *limit, unsigned long mm_flags) +static bool elf_fdpic_dump_segments(struct coredump_params *cprm)  {  	struct vm_area_struct *vma; -	int err = 0;  	for (vma = current->mm->mmap; vma; vma = vma->vm_next) {  		unsigned long addr; -		if (!maydump(vma, mm_flags)) +		if (!maydump(vma, cprm->mm_flags))  			continue; +#ifdef CONFIG_MMU  		for (addr = vma->vm_start; addr < vma->vm_end;  							addr += PAGE_SIZE) { +			bool res;  			struct page *page = get_dump_page(addr);  			if (page) {  				void *kaddr = kmap(page); -				*size += PAGE_SIZE; -				if (*size > *limit) -					err = -EFBIG; -				else if (!dump_write(file, kaddr, PAGE_SIZE)) -					err = -EIO; +				res = dump_emit(cprm, kaddr, PAGE_SIZE);  				kunmap(page);  				page_cache_release(page); -			} else if (!dump_seek(file, PAGE_SIZE)) -				err = -EFBIG; -			if (err) -				goto out; +			} else { +				res = dump_skip(cprm, PAGE_SIZE); +			} +			if (!res) +				return false;  		} -	} -out: -	return err; -} -#endif - -/* - * dump the segments for a NOMMU process - */ -#ifndef CONFIG_MMU -static int elf_fdpic_dump_segments(struct file *file, size_t *size, -			   unsigned long *limit, unsigned long mm_flags) -{ -	struct vm_area_struct *vma; - -	for (vma = current->mm->mmap; vma; vma = vma->vm_next) { -		if (!maydump(vma, mm_flags)) -			continue; - -		if ((*size += PAGE_SIZE) > *limit) -			return -EFBIG; - -		if (!dump_write(file, (void *) vma->vm_start, +#else +		if (!dump_emit(cprm, (void *) vma->vm_start,  				vma->vm_end - vma->vm_start)) -			return -EIO; +			return false; +#endif  	} - -	return 0; +	return true;  } -#endif  static size_t elf_core_vma_data_size(unsigned long mm_flags)  { @@ -1598,11 +1541,10 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  	int has_dumped = 0;  	mm_segment_t fs;  	int segs; -	size_t size = 0;  	int i;  	struct vm_area_struct *vma;  	struct elfhdr *elf = NULL; -	loff_t offset = 0, dataoff, foffset; +	loff_t offset = 0, dataoff;  	int numnote;  	struct memelfnote *notes = NULL;  	struct elf_prstatus *prstatus = NULL;	/* NT_PRSTATUS */ @@ -1619,6 +1561,8 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  	struct elf_shdr *shdr4extnum = NULL;  	Elf_Half e_phnum;  	elf_addr_t e_shoff; +	struct core_thread *ct; +	struct elf_thread_status *tmp;  	/*  	 * We no longer stop all VM operations. @@ -1654,32 +1598,27 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  		goto cleanup;  #endif -	if (cprm->signr) { -		struct core_thread *ct; -		struct elf_thread_status *tmp; - -		for (ct = current->mm->core_state->dumper.next; -						ct; ct = ct->next) { -			tmp = kzalloc(sizeof(*tmp), GFP_KERNEL); -			if (!tmp) -				goto cleanup; +	for (ct = current->mm->core_state->dumper.next; +					ct; ct = ct->next) { +		tmp = kzalloc(sizeof(*tmp), GFP_KERNEL); +		if (!tmp) +			goto cleanup; -			tmp->thread = ct->task; -			list_add(&tmp->list, &thread_list); -		} +		tmp->thread = ct->task; +		list_add(&tmp->list, &thread_list); +	} -		list_for_each(t, &thread_list) { -			struct elf_thread_status *tmp; -			int sz; +	list_for_each(t, &thread_list) { +		struct elf_thread_status *tmp; +		int sz; -			tmp = list_entry(t, struct elf_thread_status, list); -			sz = elf_dump_thread_status(cprm->signr, tmp); -			thread_status_size += sz; -		} +		tmp = list_entry(t, struct elf_thread_status, list); +		sz = elf_dump_thread_status(cprm->siginfo->si_signo, tmp); +		thread_status_size += sz;  	}  	/* now collect the dump for the current */ -	fill_prstatus(prstatus, current, cprm->signr); +	fill_prstatus(prstatus, current, cprm->siginfo->si_signo);  	elf_core_copy_regs(&prstatus->pr_reg, cprm->regs);  	segs = current->mm->map_count; @@ -1697,8 +1636,6 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  	fill_elf_fdpic_header(elf, e_phnum);  	has_dumped = 1; -	current->flags |= PF_DUMPCORE; -  	/*  	 * Set up the notes in similar form to SVR4 core dumps made  	 * with info from their /proc. @@ -1735,7 +1672,6 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  	offset += sizeof(*elf);				/* Elf header */  	offset += segs * sizeof(struct elf_phdr);	/* Program headers */ -	foffset = offset;  	/* Write notes phdr entry */  	{ @@ -1770,13 +1706,10 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  	offset = dataoff; -	size += sizeof(*elf); -	if (size > cprm->limit || !dump_write(cprm->file, elf, sizeof(*elf))) +	if (!dump_emit(cprm, elf, sizeof(*elf)))  		goto end_coredump; -	size += sizeof(*phdr4note); -	if (size > cprm->limit -	    || !dump_write(cprm->file, phdr4note, sizeof(*phdr4note))) +	if (!dump_emit(cprm, phdr4note, sizeof(*phdr4note)))  		goto end_coredump;  	/* write program headers for segments dump */ @@ -1800,18 +1733,16 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  			phdr.p_flags |= PF_X;  		phdr.p_align = ELF_EXEC_PAGESIZE; -		size += sizeof(phdr); -		if (size > cprm->limit -		    || !dump_write(cprm->file, &phdr, sizeof(phdr))) +		if (!dump_emit(cprm, &phdr, sizeof(phdr)))  			goto end_coredump;  	} -	if (!elf_core_write_extra_phdrs(cprm->file, offset, &size, cprm->limit)) +	if (!elf_core_write_extra_phdrs(cprm, offset))  		goto end_coredump;   	/* write out the notes section */  	for (i = 0; i < numnote; i++) -		if (!writenote(notes + i, cprm->file, &foffset)) +		if (!writenote(notes + i, cprm))  			goto end_coredump;  	/* write out the thread status notes section */ @@ -1820,25 +1751,21 @@ static int elf_fdpic_core_dump(struct coredump_params *cprm)  				list_entry(t, struct elf_thread_status, list);  		for (i = 0; i < tmp->num_notes; i++) -			if (!writenote(&tmp->notes[i], cprm->file, &foffset)) +			if (!writenote(&tmp->notes[i], cprm))  				goto end_coredump;  	} -	if (!dump_seek(cprm->file, dataoff - foffset)) +	if (!dump_skip(cprm, dataoff - cprm->written))  		goto end_coredump; -	if (elf_fdpic_dump_segments(cprm->file, &size, &cprm->limit, -				    cprm->mm_flags) < 0) +	if (!elf_fdpic_dump_segments(cprm))  		goto end_coredump; -	if (!elf_core_write_extra_data(cprm->file, &size, cprm->limit)) +	if (!elf_core_write_extra_data(cprm))  		goto end_coredump;  	if (e_phnum == PN_XNUM) { -		size += sizeof(*shdr4extnum); -		if (size > cprm->limit -		    || !dump_write(cprm->file, shdr4extnum, -				   sizeof(*shdr4extnum))) +		if (!dump_emit(cprm, shdr4extnum, sizeof(*shdr4extnum)))  			goto end_coredump;  	} @@ -1864,6 +1791,7 @@ cleanup:  	kfree(psinfo);  	kfree(notes);  	kfree(fpu); +	kfree(shdr4extnum);  #ifdef ELF_CORE_COPY_XFPREGS  	kfree(xfpu);  #endif  | 
