| // SPDX-License-Identifier: (GPL-2.0+ OR MIT) |
| /* |
| * drivers/amlogic/memory_ext/ram_dump.c |
| * |
| * Copyright (C) 2017 Amlogic, Inc. All rights reserved. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for |
| * more details. |
| * |
| */ |
| #define DEBUG |
| #include <linux/version.h> |
| #include <linux/types.h> |
| #include <linux/fs.h> |
| #include <linux/init.h> |
| #include <linux/device.h> |
| #include <linux/platform_device.h> |
| #include <linux/mutex.h> |
| #include <linux/io.h> |
| #include <linux/module.h> |
| #include <linux/kernel.h> |
| #include <linux/slab.h> |
| #include <linux/mm.h> |
| #include <linux/reboot.h> |
| #include <linux/memblock.h> |
| #include <linux/sched/clock.h> |
| #include <linux/amlogic/ramdump.h> |
| #include <linux/amlogic/reboot.h> |
| #include <linux/arm-smccc.h> |
| #include <linux/of.h> |
| #include <linux/highmem.h> |
| #include <linux/syscalls.h> |
| #include <linux/fs_struct.h> |
| #include <linux/mount.h> |
| #include <linux/namei.h> |
| #include <linux/capability.h> |
| #include <asm/cacheflush.h> |
| #include <linux/amlogic/gki_module.h> |
| |
| static unsigned long ramdump_base; |
| static unsigned long ramdump_size; |
| static bool ramdump_disable = 1; |
| |
| #define WAIT_TIMEOUT (40ULL * 1000 * 1000 * 1000) |
| //#define SAVE_CRASH_TO_ANDROID_DATA 1 |
| #define SAVE_DATA_BY_INIT_RC_SHELL 1 |
| //#define SAVE_DATA_BY_RAMDUMP_DRIVER 1 |
| |
| struct ramdump { |
| unsigned long mem_size; |
| unsigned long mem_base; |
| #ifdef SAVE_DATA_BY_INIT_RC_SHELL |
| //void __iomem *mem_base; |
| struct mutex lock; |
| struct kobject *kobj; |
| #else |
| unsigned long long tick; |
| void *mnt_buf; |
| const char *storage_device; |
| #endif |
| struct delayed_work work; |
| int disable; |
| }; |
| |
| static struct ramdump *ram; |
| |
| static int early_ramdump_para(char *buf) |
| { |
| int ret; |
| |
| if (!buf) |
| return -EINVAL; |
| |
| pr_info("%s:%s\n", __func__, buf); |
| if (strcmp(buf, "disabled") == 0) { |
| ramdump_disable = 1; |
| } else { |
| ret = sscanf(buf, "%lx,%lx", &ramdump_base, &ramdump_size); |
| if (ret != 2) { |
| pr_err("invalid boot args\n"); |
| ramdump_disable = 1; |
| } |
| ramdump_disable = 0; |
| pr_info("%s, base:%lx, size:%lx\n", |
| __func__, ramdump_base, ramdump_size); |
| #ifdef SAVE_CRASH_TO_ANDROID_DATA |
| ret = memblock_reserve(ramdump_base, PAGE_ALIGN(ramdump_size)); |
| if (ret < 0) { |
| pr_info("%s, reserve memblock %lx - %lx failed\n", |
| __func__, ramdump_base, |
| ramdump_base + PAGE_ALIGN(ramdump_size)); |
| ramdump_disable = 1; |
| } else { |
| pr_info("%s, reserve memblock %lx - %lx OK\n", |
| __func__, ramdump_base, |
| ramdump_base + PAGE_ALIGN(ramdump_size)); |
| } |
| #endif |
| } |
| return 0; |
| } |
| |
| early_param("ramdump", early_ramdump_para); |
| |
| #ifdef SAVE_DATA_BY_INIT_RC_SHELL |
| static ssize_t ramdump_bin_read(struct file *filp, struct kobject *kobj, |
| struct bin_attribute *attr, |
| char *buf, loff_t off, size_t count) |
| { |
| void *p = NULL; |
| |
| if (!ram->mem_base || off >= ram->mem_size) { |
| pr_info("%s, crash sysfsnode data err.\n", __func__); |
| return 0; |
| } |
| |
| if (off + count > ram->mem_size) |
| count = ram->mem_size - off; |
| |
| p = (void *)phys_to_virt(ram->mem_base + off); |
| |
| mutex_lock(&ram->lock); |
| memcpy(buf, p, count); |
| mutex_unlock(&ram->lock); |
| |
| /* debug when read end */ |
| if (off + count >= ram->mem_size) |
| pr_info("%s, p=%p %p, off:%lli, c:%zi\n", |
| __func__, buf, p, off, count); |
| |
| return count; |
| } |
| |
| int ramdump_disabled(void) |
| { |
| if (ram) |
| return ram->disable; |
| return 0; |
| } |
| EXPORT_SYMBOL(ramdump_disabled); |
| |
| static void meson_set_reboot_reason(int reboot_reason) |
| { |
| struct arm_smccc_res smccc; |
| |
| arm_smccc_smc(SET_REBOOT_REASON, |
| reboot_reason, 0, 0, 0, 0, 0, 0, &smccc); |
| } |
| |
| static ssize_t ramdump_bin_write(struct file *filp, |
| struct kobject *kobj, |
| struct bin_attribute *bin_attr, |
| char *buf, loff_t off, size_t count) |
| { |
| if (ram->mem_base && !strncmp("reboot", buf, 6)) |
| kernel_restart("RAM-DUMP finished\n"); |
| |
| if (!strncmp("disable", buf, 7)) { |
| ram->disable = 1; |
| meson_set_reboot_reason(MESON_NORMAL_BOOT); |
| } |
| if (!strncmp("enable", buf, 6)) { |
| ram->disable = 0; |
| meson_set_reboot_reason(MESON_KERNEL_PANIC); |
| } |
| |
| return count; |
| } |
| |
| static struct bin_attribute ramdump_attr = { |
| .attr = { |
| .name = "compmsg", |
| .mode = 0664, |
| }, |
| .read = ramdump_bin_read, |
| .write = ramdump_bin_write, |
| }; |
| #else |
| #ifdef CONFIG_ARM64 |
| void ramdump_sync_data(void) |
| { |
| /* |
| * back port from old kernel version for function |
| * flush_cache_all(), we need it for ram dump |
| */ |
| asm volatile |
| ("mov x12, x30\n" |
| "dsb sy\n" |
| "mrs x0, clidr_el1\n" |
| "and x3, x0, #0x7000000\n" |
| "lsr x3, x3, #23\n" |
| "cbz x3, finished\n" |
| "mov x10, #0\n" |
| "loop1:\n" |
| "add x2, x10, x10, lsr #1\n" |
| "lsr x1, x0, x2\n" |
| "and x1, x1, #7\n" |
| "cmp x1, #2\n" |
| "b.lt skip\n" |
| "mrs x9, daif\n" |
| "msr daifset, #2\n" |
| "msr csselr_el1, x10\n" |
| "isb\n" |
| "mrs x1, ccsidr_el1\n" |
| "msr daif, x9\n" |
| "and x2, x1, #7\n" |
| "add x2, x2, #4\n" |
| "mov x4, #0x3ff\n" |
| "and x4, x4, x1, lsr #3\n" |
| "clz w5, w4\n" |
| "mov x7, #0x7fff\n" |
| "and x7, x7, x1, lsr #13\n" |
| "loop2:\n" |
| "mov x9, x4\n" |
| "loop3:\n" |
| "lsl x6, x9, x5\n" |
| "orr x11, x10, x6\n" |
| "lsl x6, x7, x2\n" |
| "orr x11, x11, x6\n" |
| "dc cisw, x11\n" |
| "subs x9, x9, #1\n" |
| "b.ge loop3\n" |
| "subs x7, x7, #1\n" |
| "b.ge loop2\n" |
| "skip:\n" |
| "add x10, x10, #2\n" |
| "cmp x3, x10\n" |
| "b.gt loop1\n" |
| "finished:\n" |
| "mov x10, #0\n" |
| "msr csselr_el1, x10\n" |
| "dsb sy\n" |
| "isb\n" |
| "mov x0, #0\n" |
| "ic ialluis\n" |
| "ret x12\n"); |
| } |
| #else |
| void ramdump_sync_data(void) |
| { |
| flush_cache_all(); |
| } |
| #endif |
| |
| #ifdef CONFIG_64BIT |
| static void free_reserved_highmem(unsigned long start, unsigned long end) |
| { |
| } |
| #else |
| static void free_reserved_highmem(unsigned long start, unsigned long end) |
| { |
| for (; start < end; ) { |
| free_highmem_page(phys_to_page(start)); |
| start += PAGE_SIZE; |
| } |
| } |
| #endif |
| |
| static void free_reserved_mem(unsigned long start, unsigned long size) |
| { |
| unsigned long end = PAGE_ALIGN(start + size); |
| struct page *page, *epage; |
| |
| page = phys_to_page(start); |
| if (PageHighMem(page)) { |
| free_reserved_highmem(start, end); |
| } else { |
| epage = phys_to_page(end); |
| if (!PageHighMem(epage)) { |
| free_reserved_area(__va(start), |
| __va(end), 0, "ramdump"); |
| } else { |
| /* reserved area cross zone */ |
| struct zone *zone; |
| unsigned long bound; |
| |
| zone = page_zone(page); |
| bound = zone_end_pfn(zone); |
| free_reserved_area(__va(start), |
| __va(bound << PAGE_SHIFT), |
| 0, "ramdump"); |
| zone = page_zone(epage); |
| bound = zone->zone_start_pfn; |
| free_reserved_highmem(bound << PAGE_SHIFT, end); |
| } |
| } |
| } |
| |
| static int check_storage_mounted(char **root) |
| { |
| int fd, cnt, ret = 0; |
| char mnt_dev[64] = {}, *mnt_ptr, *root_dir; |
| |
| pr_info("%s ?\n", __func__); |
| fd = ksys_open("/proc/mounts", O_RDONLY, 0); |
| if (!fd) { |
| pr_info("%s, open mounts failed:%d\n", __func__, fd); |
| return -EINVAL; |
| } |
| cnt = ksys_read(fd, ram->mnt_buf, PAGE_SIZE); |
| if (cnt < 0) { |
| pr_info("%s, read mounts failed:%d\n", __func__, cnt); |
| ret = -ENODEV; |
| goto exit; |
| } |
| pr_info("read:%d, %s\n", cnt, (char *)ram->mnt_buf); |
| /* for android */ |
| sprintf(mnt_dev, "/dev/block/%s", ram->storage_device); |
| mnt_ptr = strstr((char *)ram->mnt_buf, mnt_dev); |
| if (mnt_ptr) { |
| pr_debug("%s, find %s in buffer, ptr:%p\n", |
| __func__, mnt_dev, mnt_ptr); |
| root_dir = strstr(mnt_ptr, " "); |
| root_dir++; |
| *root = root_dir; |
| pr_info("mount:%s root:%s\n", mnt_ptr, root_dir); |
| } else { |
| /* for build root */ |
| memset(mnt_dev, 0, sizeof(mnt_dev)); |
| sprintf(mnt_dev, "/dev/%s", ram->storage_device); |
| mnt_ptr = strstr((char *)ram->mnt_buf, mnt_dev); |
| if (mnt_ptr) { |
| pr_debug("%s, find %s in buffer, ptr:%p\n", |
| __func__, mnt_dev, mnt_ptr); |
| root_dir = strstr(mnt_ptr, " "); |
| root_dir++; |
| *root = root_dir; |
| pr_info("mount:%s root:%s\n", mnt_ptr, root_dir); |
| } else { |
| ret = -ENODEV; |
| } |
| } |
| exit: |
| ksys_close(fd); |
| return ret; |
| } |
| |
| static size_t save_data(int fd) |
| { |
| unsigned long saved = 0, off = 0, s = 0, e; |
| void *buffer; |
| int block = (1 << (PAGE_SHIFT + MAX_ORDER - 1)), wsize = 0, ret, i; |
| struct vm_struct *area; |
| struct page *page, **pages = NULL; |
| |
| area = get_vm_area(block, VM_ALLOC); |
| if (!area) { |
| pr_err("%s, get vma failed\n", __func__); |
| return -ENOMEM; |
| } |
| |
| pages = kzalloc(sizeof(unsigned long) * MAX_ORDER_NR_PAGES, GFP_KERNEL); |
| if (!pages) |
| goto out; |
| |
| buffer = area->addr; |
| while (saved < ram->mem_size) { |
| s = ram->mem_size - saved; |
| if (s >= block) |
| wsize = block; |
| else |
| wsize = s; |
| |
| s = ram->mem_base + off; |
| e = s + PAGE_ALIGN(wsize); |
| |
| page = phys_to_page(s); |
| for (i = 0; i < MAX_ORDER_NR_PAGES; i++) { |
| pages[i] = page; |
| page++; |
| } |
| ret = map_kernel_range_noflush((unsigned long)buffer, |
| PAGE_ALIGN(wsize), |
| PAGE_KERNEL, |
| pages); |
| if (!ret) { |
| pr_err("map page:%lx failed\n", page_to_pfn(page)); |
| goto out; |
| } |
| ret = ksys_write(fd, buffer, wsize); |
| if (ret != wsize) { |
| unmap_kernel_range((unsigned long)buffer, |
| PAGE_ALIGN(wsize)); |
| pr_err("%s, write failed\n", __func__); |
| goto out; |
| } |
| unmap_kernel_range((unsigned long)buffer, PAGE_ALIGN(wsize)); |
| free_reserved_mem(s, PAGE_ALIGN(wsize)); |
| saved += wsize; |
| off += wsize; |
| pr_debug("%s, write %08lx, size:%08x, saved:%08lx, off:%lx\n", |
| __func__, s, wsize, saved, off); |
| } |
| out: |
| free_vm_area(area); |
| kfree(pages); |
| pr_info("%s, write %08lx, size:%08x, saved:%08lx, off:%lx\n", |
| __func__, s, wsize, saved, off); |
| return saved; |
| } |
| |
| #define OPEN_FLAGS (O_WRONLY | O_CREAT | O_DSYNC | O_TRUNC) |
| |
| static void wait_to_save(struct work_struct *work) |
| { |
| char *root; |
| int fd, ret = 0; |
| int need_reboot = 0; |
| |
| pr_info("%s, copy ramdump file to flash.\n", __func__); |
| if ((sched_clock() - ram->tick) >= WAIT_TIMEOUT) { |
| pr_err("can't find mounted device, free saved data\n"); |
| need_reboot = 3; |
| goto exit; |
| } |
| |
| if (!check_storage_mounted(&root)) { |
| char wname[64] = {}, *next_token; |
| |
| pr_info("%s,%d: exit.\n", __func__, __LINE__); |
| /* write compressed data to storage device */ |
| next_token = strstr(root, " "); |
| if (next_token) |
| *next_token = '\0'; |
| sprintf(wname, "%s/crashdump-1.bin", root); |
| fd = ksys_open(wname, OPEN_FLAGS, 0644); |
| if (fd < 0) { |
| pr_info("open %s failed:%d\n", wname, fd); |
| need_reboot = 3; |
| goto exit; |
| } |
| pr_info("%s,%d: exit.\n", __func__, __LINE__); |
| ret = save_data(fd); |
| if (ret != ram->mem_size) { |
| pr_err("write size %d not match %ld\n", |
| ret, ram->mem_size); |
| } |
| ksys_close(fd); |
| ksys_sync(); |
| need_reboot = 1; |
| } else { |
| pr_info("%s,%d: exit.\n", __func__, __LINE__); |
| schedule_delayed_work(&ram->work, 500); |
| } |
| |
| exit: |
| /* Nomatter what happened, reboot must be done in this function */ |
| if (need_reboot) { |
| if (need_reboot & 0x1) |
| kfree(ram->mnt_buf); |
| if (need_reboot & 0x02) |
| free_reserved_mem(ram->mem_base, ram->mem_size); |
| kernel_restart("RAM-DUMP finished"); |
| } |
| } |
| #endif |
| |
| /* |
| * clear memory to avoid large amount of memory not used. |
| * for random data, it's hard to compress |
| */ |
| static void lazy_clear_work(struct work_struct *work) |
| { |
| struct page *page; |
| struct list_head head, *pos, *next; |
| void *virt; |
| int order; |
| gfp_t flags = __GFP_NORETRY | |
| __GFP_NOWARN | |
| __GFP_MOVABLE; |
| unsigned long clear = 0, size = 0, free = 0, tick; |
| |
| INIT_LIST_HEAD(&head); |
| order = MAX_ORDER - 1; |
| tick = sched_clock(); |
| do { |
| page = alloc_pages(flags, order); |
| if (page) { |
| list_add(&page->lru, &head); |
| virt = page_address(page); |
| size = (1 << order) * PAGE_SIZE; |
| memset(virt, 0, size); |
| clear += size; |
| } |
| } while (page); |
| tick = sched_clock() - tick; |
| |
| list_for_each_safe(pos, next, &head) { |
| page = list_entry(pos, struct page, lru); |
| list_del(&page->lru); |
| __free_pages(page, order); |
| free += size; |
| } |
| pr_info("clear:%lx, free:%lx, tick:%ld us\n", |
| clear, free, tick / 1000); |
| } |
| |
| static int __init ramdump_probe(struct platform_device *pdev) |
| { |
| unsigned long total_mem; |
| struct resource *res; |
| unsigned int dump_set; |
| void __iomem *base; |
| void *vaddr = NULL; |
| int ret = 0; |
| |
| total_mem = get_num_physpages() << PAGE_SHIFT; |
| pr_info("Total Memory:[%lx]\n", total_mem); |
| |
| ram = kzalloc(sizeof(*ram), GFP_KERNEL); |
| if (!ram) |
| return -ENOMEM; |
| |
| if (ramdump_disable) |
| ram->disable = 1; |
| |
| ram->mem_base = 0; |
| ram->mem_size = ramdump_size; |
| if (!ramdump_base || !ramdump_size) { |
| pr_info("NO valid ramdump args:%lx %lx\n", |
| ramdump_base, ramdump_size); |
| } else { |
| pr_info("%s, memremap start, paddr area: 0x%08lx - 0x%08lx\n", |
| __func__, ramdump_base, ramdump_base + PAGE_ALIGN(ramdump_size)); |
| //vaddr = ioremap_cache(ramdump_base, PAGE_ALIGN(ramdump_size)); |
| vaddr = memremap(ramdump_base, PAGE_ALIGN(ramdump_size), MEMREMAP_WB); |
| if (vaddr) |
| ram->mem_base = (unsigned long)vaddr; |
| |
| pr_info("%s, memremap end, vaddr_base:%lx, size:%lx\n", |
| __func__, ram->mem_base, ram->mem_size); |
| } |
| |
| if (!ram->disable) { |
| if (!ram->mem_base) { /* No compressed data */ |
| INIT_DELAYED_WORK(&ram->work, lazy_clear_work); |
| schedule_delayed_work(&ram->work, msecs_to_jiffies(100)); |
| } else { /* with compressed data */ |
| #ifdef SAVE_DATA_BY_INIT_RC_SHELL |
| pr_info("%s, SAVE_DATA_BY_INIT_RC_SHELL\n", __func__); |
| ram->kobj = kobject_create_and_add("mdump", kernel_kobj); |
| if (!ram->kobj) { |
| pr_err("%s, create sysfs /mdump failed\n", __func__); |
| goto err; |
| } |
| #ifdef SAVE_CRASH_TO_ANDROID_DATA |
| ramdump_attr.size = ram->mem_size; |
| pr_info("%s, creat /sys/kernel/mdump/compmsg\n", __func__); |
| if (sysfs_create_bin_file(ram->kobj, &ramdump_attr)) { |
| pr_err("%s, create sysfs compmsg failed\n", __func__); |
| goto err1; |
| } |
| #endif /* end SAVE_CRASH_TO_ANDROID_DATA */ |
| #else /* SAVE_DATA_BY_RAMDUMP_DRIVER */ |
| struct device_node *np; |
| const char *dev_name = NULL; |
| |
| np = pdev->dev.of_node; |
| ret = of_property_read_string(np, "store_device", &dev_name); |
| if (!ret) { |
| ram->storage_device = dev_name; |
| pr_info("%s, storage device:%s\n", __func__, dev_name); |
| } |
| pr_info("%s, SAVE_DATA_BY_RAMDUMP_DRIVER\n", __func__); |
| INIT_DELAYED_WORK(&ram->work, wait_to_save); |
| ram->tick = sched_clock(); |
| ram->mnt_buf = kmalloc(PAGE_SIZE, GFP_KERNEL); |
| WARN_ON(!ram->mnt_buf); |
| #endif /* end SAVE_DATA_BY_INIT_RC_SHELL */ |
| } |
| |
| /* if ramdump is disabled in env, no need to set sticky reg */ |
| res = platform_get_resource_byname(pdev, IORESOURCE_MEM, |
| "SYSCTRL_STICKY_REG6"); |
| if (res) { |
| base = devm_ioremap(&pdev->dev, res->start, |
| res->end - res->start); |
| if (!base) { |
| pr_err("%s, map reg failed\n", __func__); |
| goto err; |
| } |
| dump_set = readl(base); |
| dump_set &= ~RAMDUMP_STICKY_DATA_MASK; |
| dump_set |= ((total_mem >> 20) | AMLOGIC_KERNEL_BOOTED); |
| writel(dump_set, base); |
| pr_info("%s, set sticky to %x\n", __func__, dump_set); |
| } |
| } |
| return ret; |
| #ifdef SAVE_CRASH_TO_ANDROID_DATA |
| err1: |
| kobject_put(ram->kobj); |
| #endif |
| |
| err: |
| kfree(ram); |
| |
| return -EINVAL; |
| } |
| |
| static int ramdump_remove(struct platform_device *pdev) |
| { |
| #ifdef SAVE_DATA_BY_INIT_RC_SHELL |
| sysfs_remove_bin_file(ram->kobj, &ramdump_attr); |
| iounmap((void *)ram->mem_base); |
| kobject_put(ram->kobj); |
| #endif |
| kfree(ram); |
| return 0; |
| } |
| |
| #ifdef CONFIG_OF |
| static const struct of_device_id ramdump_dt_match[] = { |
| { |
| .compatible = "amlogic, ram_dump", |
| }, |
| {}, |
| }; |
| #endif |
| |
| static struct platform_driver ramdump_driver = { |
| .driver = { |
| .name = "mdump", |
| .owner = THIS_MODULE, |
| #ifdef CONFIG_OF |
| .of_match_table = ramdump_dt_match, |
| #endif |
| }, |
| .remove = ramdump_remove, |
| }; |
| |
| int __init ramdump_init(void) |
| { |
| int ret; |
| |
| ret = platform_driver_probe(&ramdump_driver, ramdump_probe); |
| |
| return ret; |
| } |
| |
| void __exit ramdump_uninit(void) |
| { |
| platform_driver_unregister(&ramdump_driver); |
| } |
| |