cregit-Linux how code gets into the kernel

Release 4.14 arch/um/kernel/skas/uaccess.c

/*
 * Copyright (C) 2002 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
 * Licensed under the GPL
 */

#include <linux/err.h>
#include <linux/highmem.h>
#include <linux/mm.h>
#include <linux/module.h>
#include <linux/sched.h>
#include <asm/current.h>
#include <asm/page.h>
#include <asm/pgtable.h>
#include <kern_util.h>
#include <os.h>


pte_t *virt_to_pte(struct mm_struct *mm, unsigned long addr) { pgd_t *pgd; pud_t *pud; pmd_t *pmd; if (mm == NULL) return NULL; pgd = pgd_offset(mm, addr); if (!pgd_present(*pgd)) return NULL; pud = pud_offset(pgd, addr); if (!pud_present(*pud)) return NULL; pmd = pmd_offset(pud, addr); if (!pmd_present(*pmd)) return NULL; return pte_offset_kernel(pmd, addr); }

Contributors

PersonTokensPropCommitsCommitProp
Jeff Dike9891.59%266.67%
Paolo 'Blaisorblade' Giarrusso98.41%133.33%
Total107100.00%3100.00%


static pte_t *maybe_map(unsigned long virt, int is_write) { pte_t *pte = virt_to_pte(current->mm, virt); int err, dummy_code; if ((pte == NULL) || !pte_present(*pte) || (is_write && !pte_write(*pte))) { err = handle_page_fault(virt, 0, is_write, 1, &dummy_code); if (err) return NULL; pte = virt_to_pte(current->mm, virt); } if (!pte_present(*pte)) pte = NULL; return pte; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso7265.45%250.00%
Jeff Dike3834.55%250.00%
Total110100.00%4100.00%


static int do_op_one_page(unsigned long addr, int len, int is_write, int (*op)(unsigned long addr, int len, void *arg), void *arg) { jmp_buf buf; struct page *page; pte_t *pte; int n, faulted; pte = maybe_map(addr, is_write); if (pte == NULL) return -1; page = pte_page(*pte); addr = (unsigned long) kmap_atomic(page) + (addr & ~PAGE_MASK); current->thread.fault_catcher = &buf; faulted = UML_SETJMP(&buf); if (faulted == 0) n = (*op)(addr, len, arg); else n = -1; current->thread.fault_catcher = NULL; kunmap_atomic((void *)addr); return n; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso10665.03%240.00%
Jeff Dike5231.90%240.00%
Cesar Eduardo Barros53.07%120.00%
Total163100.00%5100.00%


static long buffer_op(unsigned long addr, int len, int is_write, int (*op)(unsigned long, int, void *), void *arg) { long size, remain, n; size = min(PAGE_ALIGN(addr) - addr, (unsigned long) len); remain = len; n = do_op_one_page(addr, size, is_write, op, arg); if (n != 0) { remain = (n < 0 ? remain : 0); goto out; } addr += size; remain -= size; if (remain == 0) goto out; while (addr < ((addr + remain) & PAGE_MASK)) { n = do_op_one_page(addr, PAGE_SIZE, is_write, op, arg); if (n != 0) { remain = (n < 0 ? remain : 0); goto out; } addr += PAGE_SIZE; remain -= PAGE_SIZE; } if (remain == 0) goto out; n = do_op_one_page(addr, remain, is_write, op, arg); if (n != 0) { remain = (n < 0 ? remain : 0); goto out; } return 0; out: return remain; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso15866.95%233.33%
Jeff Dike4820.34%233.33%
Bodo Stroesser2811.86%116.67%
Richard Weinberger20.85%116.67%
Total236100.00%6100.00%


static int copy_chunk_from_user(unsigned long from, int len, void *arg) { unsigned long *to_ptr = arg, to = *to_ptr; memcpy((void *) to, (void *) from, len); *to_ptr += len; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso54100.00%1100.00%
Total54100.00%1100.00%


unsigned long raw_copy_from_user(void *to, const void __user *from, unsigned long n) { if (uaccess_kernel()) { memcpy(to, (__force void*)from, n); return 0; } return buffer_op((unsigned long) from, n, 0, copy_chunk_from_user, &to); }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso5180.95%120.00%
Jeff Dike69.52%120.00%
Al Viro34.76%240.00%
Richard Weinberger34.76%120.00%
Total63100.00%5100.00%

EXPORT_SYMBOL(raw_copy_from_user);
static int copy_chunk_to_user(unsigned long to, int len, void *arg) { unsigned long *from_ptr = arg, from = *from_ptr; memcpy((void *) to, (void *) from, len); *from_ptr += len; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso54100.00%1100.00%
Total54100.00%1100.00%


unsigned long raw_copy_to_user(void __user *to, const void *from, unsigned long n) { if (uaccess_kernel()) { memcpy((__force void *) to, from, n); return 0; } return buffer_op((unsigned long) to, n, 1, copy_chunk_to_user, &from); }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso5180.95%120.00%
Jeff Dike69.52%120.00%
Al Viro34.76%240.00%
Richard Weinberger34.76%120.00%
Total63100.00%5100.00%

EXPORT_SYMBOL(raw_copy_to_user);
static int strncpy_chunk_from_user(unsigned long from, int len, void *arg) { char **to_ptr = arg, *to = *to_ptr; int n; strncpy(to, (void *) from, len); n = strnlen(to, len); *to_ptr += n; if (n < len) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso72100.00%1100.00%
Total72100.00%1100.00%


long __strncpy_from_user(char *dst, const char __user *src, long count) { long n; char *ptr = dst; if (uaccess_kernel()) { strncpy(dst, (__force void *) src, count); return strnlen(dst, count); } n = buffer_op((unsigned long) src, count, 0, strncpy_chunk_from_user, &ptr); if (n != 0) return -EFAULT; return strnlen(dst, count); }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso8388.30%125.00%
Jeff Dike66.38%125.00%
Richard Weinberger44.26%125.00%
Al Viro11.06%125.00%
Total94100.00%4100.00%

EXPORT_SYMBOL(__strncpy_from_user);
static int clear_chunk(unsigned long addr, int len, void *unused) { memset((void *) addr, 0, len); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso33100.00%1100.00%
Total33100.00%1100.00%


unsigned long __clear_user(void __user *mem, unsigned long len) { if (uaccess_kernel()) { memset((__force void*)mem, 0, len); return 0; } return buffer_op((unsigned long) mem, len, 1, clear_chunk, NULL); }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso4782.46%120.00%
Jeff Dike712.28%240.00%
Richard Weinberger23.51%120.00%
Al Viro11.75%120.00%
Total57100.00%5100.00%

EXPORT_SYMBOL(__clear_user);
static int strnlen_chunk(unsigned long str, int len, void *arg) { int *len_ptr = arg, n; n = strnlen((void *) str, len); *len_ptr += n; if (n < len) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso55100.00%1100.00%
Total55100.00%1100.00%


long __strnlen_user(const void __user *str, long len) { int count = 0, n; if (uaccess_kernel()) return strnlen((__force char*)str, len) + 1; n = buffer_op((unsigned long) str, len, 0, strnlen_chunk, &count); if (n == 0) return count + 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso6485.33%120.00%
Jeff Dike68.00%120.00%
Richard Weinberger45.33%240.00%
Al Viro11.33%120.00%
Total75100.00%5100.00%

EXPORT_SYMBOL(__strnlen_user);

Overall Contributors

PersonTokensPropCommitsCommitProp
Paolo 'Blaisorblade' Giarrusso92371.44%419.05%
Jeff Dike27921.59%942.86%
Al Viro362.79%419.05%
Bodo Stroesser282.17%14.76%
Richard Weinberger211.63%29.52%
Cesar Eduardo Barros50.39%14.76%
Total1292100.00%21100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.