From d2e1446d81725c351dc73a03b397ce043fb18452 Mon Sep 17 00:00:00 2001 From: Meizu OpenSource Date: Mon, 15 Aug 2016 10:19:42 +0800 Subject: first commit --- tools/Makefile | 91 + tools/cgroup/.gitignore | 1 + tools/cgroup/Makefile | 11 + tools/cgroup/cgroup_event_listener.c | 82 + tools/dct/ADC_YuSu.cmp | 19 + tools/dct/Application.cmp | 52 + tools/dct/DrvGen | Bin 0 -> 3502139 bytes tools/dct/DrvGen.exe | Bin 0 -> 1270784 bytes tools/dct/EINT_YuSu.cmp | 174 + tools/dct/GPIO_YuSu.cmp | 599 +++ tools/dct/GPIO_protect.cmp | 25 + tools/dct/I2C_YuSu.cmp | 31 + tools/dct/Keypad_YuSu.cmp | 271 + tools/dct/MT6574.fig | 249 + tools/dct/MT6735.fig | 333 ++ tools/dct/MT6735M.fig | 327 ++ tools/dct/MT6753.fig | 333 ++ tools/dct/MT8173.fig | 453 ++ tools/dct/MT8590.fig | 445 ++ tools/dct/PMIC_MT6322+6333PMUMP.cmp | 263 + tools/dct/PMIC_MT6322PMUMP.cmp | 216 + tools/dct/PMIC_MT6323+6333PMUMP.cmp | 263 + tools/dct/PMIC_MT6323PMUMP.cmp | 216 + tools/dct/PMIC_MT6325PMUMP.cmp | 325 ++ tools/dct/PMIC_MT6328PMUMP.cmp | 108 + tools/dct/PMIC_MT6331+6332PMUMP.cmp | 358 ++ tools/dct/PMIC_MT6350PMUMP.cmp | 218 + tools/dct/PMIC_MT6397PMUMP.cmp | 137 + tools/dct/PMIC_NCPMU.cmp | 17 + tools/dct/UEM.cmp | 13 + tools/dct/Wizard/AST.cmp | 29 + tools/dct/Wizard/Bluetooth.cmp | 44 + tools/dct/Wizard/CMMB.cmp | 31 + tools/dct/Wizard/Camera.cmp | 127 + tools/dct/Wizard/FM.cmp | 29 + tools/dct/Wizard/GPS.cmp | 42 + tools/dct/Wizard/HW_Module.cmp | 22 + tools/dct/Wizard/I2C.cmp | 38 + tools/dct/Wizard/I2S.cmp | 31 + tools/dct/Wizard/Jogball_and_OFN.cmp | 83 + tools/dct/Wizard/JogballandOFN_old.cmp | 83 + tools/dct/Wizard/Keypad.cmp | 121 + tools/dct/Wizard/LCD.cmp | 32 + tools/dct/Wizard/MISC.cmp | 30 + tools/dct/Wizard/MotionSensor.cmp | 156 + tools/dct/Wizard/PWM.cmp | 62 + tools/dct/Wizard/SPI.cmp | 30 + tools/dct/Wizard/Sensors.cmp | 66 + tools/dct/Wizard/TouchPanel.cmp | 33 + tools/dct/Wizard/UART.cmp | 53 + tools/dct/Wizard/USB.cmp | 29 + tools/dct/Wizard/WIFI.cmp | 38 + tools/dct/Wizard/connectivity.cmp | 37 + tools/dct/Wizard/mATV.cmp | 25 + tools/dct/Wizard/mATV_old.cmp | 25 + tools/firewire/Makefile | 19 + tools/firewire/decode-fcp.c | 213 + tools/firewire/list.h | 62 + tools/firewire/nosy-dump.c | 1035 ++++ tools/firewire/nosy-dump.h | 173 + tools/hv/hv_get_dhcp_info.sh | 28 + tools/hv/hv_get_dns_info.sh | 13 + tools/hv/hv_kvp_daemon.c | 1704 ++++++ tools/hv/hv_set_ifconfig.sh | 64 + tools/hv/hv_vss_daemon.c | 249 + tools/include/tools/be_byteshift.h | 70 + tools/include/tools/le_byteshift.h | 70 + tools/lguest/.gitignore | 1 + tools/lguest/Makefile | 8 + tools/lguest/extract | 58 + tools/lguest/lguest.c | 2052 +++++++ tools/lguest/lguest.txt | 125 + tools/lib/lk/Makefile | 35 + tools/lib/lk/debugfs.c | 100 + tools/lib/lk/debugfs.h | 29 + tools/lib/traceevent/.gitignore | 1 + tools/lib/traceevent/Makefile | 319 ++ tools/lib/traceevent/event-parse.c | 5654 ++++++++++++++++++++ tools/lib/traceevent/event-parse.h | 844 +++ tools/lib/traceevent/event-utils.h | 85 + tools/lib/traceevent/parse-filter.c | 2303 ++++++++ tools/lib/traceevent/parse-utils.c | 129 + tools/lib/traceevent/trace-seq.c | 199 + tools/net/Makefile | 15 + tools/net/bpf_jit_disasm.c | 199 + tools/nfsd/inject_fault.sh | 49 + tools/perf/.gitignore | 25 + tools/perf/CREDITS | 30 + tools/perf/Documentation/Makefile | 352 ++ tools/perf/Documentation/android.txt | 78 + tools/perf/Documentation/asciidoc.conf | 91 + tools/perf/Documentation/examples.txt | 225 + tools/perf/Documentation/jit-interface.txt | 15 + tools/perf/Documentation/manpage-1.72.xsl | 14 + tools/perf/Documentation/manpage-base.xsl | 35 + tools/perf/Documentation/manpage-bold-literal.xsl | 17 + tools/perf/Documentation/manpage-normal.xsl | 13 + tools/perf/Documentation/manpage-suppress-sp.xsl | 21 + tools/perf/Documentation/perf-annotate.txt | 101 + tools/perf/Documentation/perf-archive.txt | 22 + tools/perf/Documentation/perf-bench.txt | 192 + tools/perf/Documentation/perf-buildid-cache.txt | 40 + tools/perf/Documentation/perf-buildid-list.txt | 43 + tools/perf/Documentation/perf-diff.txt | 133 + tools/perf/Documentation/perf-evlist.txt | 38 + tools/perf/Documentation/perf-help.txt | 38 + tools/perf/Documentation/perf-inject.txt | 46 + tools/perf/Documentation/perf-kmem.txt | 47 + tools/perf/Documentation/perf-kvm.txt | 100 + tools/perf/Documentation/perf-list.txt | 118 + tools/perf/Documentation/perf-lock.txt | 66 + tools/perf/Documentation/perf-mem.txt | 48 + tools/perf/Documentation/perf-probe.txt | 205 + tools/perf/Documentation/perf-record.txt | 193 + tools/perf/Documentation/perf-report.txt | 215 + tools/perf/Documentation/perf-sched.txt | 55 + tools/perf/Documentation/perf-script-perl.txt | 216 + tools/perf/Documentation/perf-script-python.txt | 620 +++ tools/perf/Documentation/perf-script.txt | 209 + tools/perf/Documentation/perf-stat.txt | 155 + tools/perf/Documentation/perf-test.txt | 32 + tools/perf/Documentation/perf-timechart.txt | 46 + tools/perf/Documentation/perf-top.txt | 193 + tools/perf/Documentation/perf-trace.txt | 59 + tools/perf/Documentation/perf.txt | 24 + tools/perf/Documentation/perfconfig.example | 29 + tools/perf/MANIFEST | 32 + tools/perf/Makefile | 1217 +++++ tools/perf/arch/arm/Makefile | 4 + tools/perf/arch/arm/util/dwarf-regs.c | 64 + tools/perf/arch/common.c | 212 + tools/perf/arch/common.h | 10 + tools/perf/arch/powerpc/Makefile | 5 + tools/perf/arch/powerpc/util/dwarf-regs.c | 88 + tools/perf/arch/powerpc/util/header.c | 36 + tools/perf/arch/s390/Makefile | 4 + tools/perf/arch/s390/util/dwarf-regs.c | 22 + tools/perf/arch/sh/Makefile | 4 + tools/perf/arch/sh/util/dwarf-regs.c | 55 + tools/perf/arch/sparc/Makefile | 4 + tools/perf/arch/sparc/util/dwarf-regs.c | 43 + tools/perf/arch/x86/Makefile | 8 + tools/perf/arch/x86/include/perf_regs.h | 80 + tools/perf/arch/x86/util/dwarf-regs.c | 75 + tools/perf/arch/x86/util/header.c | 59 + tools/perf/arch/x86/util/unwind.c | 111 + tools/perf/bash_completion | 62 + tools/perf/bench/bench.h | 44 + tools/perf/bench/mem-memcpy-arch.h | 12 + tools/perf/bench/mem-memcpy-x86-64-asm-def.h | 12 + tools/perf/bench/mem-memcpy-x86-64-asm.S | 12 + tools/perf/bench/mem-memcpy.c | 303 ++ tools/perf/bench/mem-memset-arch.h | 12 + tools/perf/bench/mem-memset-x86-64-asm-def.h | 12 + tools/perf/bench/mem-memset-x86-64-asm.S | 13 + tools/perf/bench/mem-memset.c | 297 + tools/perf/bench/numa.c | 1731 ++++++ tools/perf/bench/sched-messaging.c | 336 ++ tools/perf/bench/sched-pipe.c | 127 + tools/perf/builtin-annotate.c | 359 ++ tools/perf/builtin-bench.c | 267 + tools/perf/builtin-buildid-cache.c | 221 + tools/perf/builtin-buildid-list.c | 106 + tools/perf/builtin-diff.c | 619 +++ tools/perf/builtin-evlist.c | 61 + tools/perf/builtin-help.c | 479 ++ tools/perf/builtin-inject.c | 461 ++ tools/perf/builtin-kmem.c | 781 +++ tools/perf/builtin-kvm.c | 1047 ++++ tools/perf/builtin-list.c | 62 + tools/perf/builtin-lock.c | 997 ++++ tools/perf/builtin-mem.c | 242 + tools/perf/builtin-probe.c | 470 ++ tools/perf/builtin-record.c | 1070 ++++ tools/perf/builtin-report.c | 962 ++++ tools/perf/builtin-sched.c | 1785 ++++++ tools/perf/builtin-script.c | 1542 ++++++ tools/perf/builtin-stat.c | 1552 ++++++ tools/perf/builtin-timechart.c | 1095 ++++ tools/perf/builtin-top.c | 1231 +++++ tools/perf/builtin-trace.c | 685 +++ tools/perf/builtin.h | 42 + tools/perf/command-list.txt | 27 + tools/perf/config/feature-tests.mak | 236 + tools/perf/config/utilities.mak | 194 + tools/perf/design.txt | 463 ++ tools/perf/perf-archive.sh | 47 + tools/perf/perf.c | 543 ++ tools/perf/perf.h | 233 + tools/perf/python/twatch.py | 41 + tools/perf/scripts/perl/Perf-Trace-Util/Context.c | 135 + tools/perf/scripts/perl/Perf-Trace-Util/Context.xs | 42 + .../perf/scripts/perl/Perf-Trace-Util/Makefile.PL | 17 + tools/perf/scripts/perl/Perf-Trace-Util/README | 59 + .../perl/Perf-Trace-Util/lib/Perf/Trace/Context.pm | 55 + .../perl/Perf-Trace-Util/lib/Perf/Trace/Core.pm | 192 + .../perl/Perf-Trace-Util/lib/Perf/Trace/Util.pm | 94 + tools/perf/scripts/perl/Perf-Trace-Util/typemap | 1 + .../perf/scripts/perl/bin/check-perf-trace-record | 2 + tools/perf/scripts/perl/bin/failed-syscalls-record | 2 + tools/perf/scripts/perl/bin/failed-syscalls-report | 10 + tools/perf/scripts/perl/bin/rw-by-file-record | 3 + tools/perf/scripts/perl/bin/rw-by-file-report | 10 + tools/perf/scripts/perl/bin/rw-by-pid-record | 2 + tools/perf/scripts/perl/bin/rw-by-pid-report | 3 + tools/perf/scripts/perl/bin/rwtop-record | 2 + tools/perf/scripts/perl/bin/rwtop-report | 20 + tools/perf/scripts/perl/bin/wakeup-latency-record | 6 + tools/perf/scripts/perl/bin/wakeup-latency-report | 3 + tools/perf/scripts/perl/check-perf-trace.pl | 106 + tools/perf/scripts/perl/failed-syscalls.pl | 42 + tools/perf/scripts/perl/rw-by-file.pl | 106 + tools/perf/scripts/perl/rw-by-pid.pl | 184 + tools/perf/scripts/perl/rwtop.pl | 203 + tools/perf/scripts/perl/wakeup-latency.pl | 107 + .../perf/scripts/python/Perf-Trace-Util/Context.c | 88 + .../python/Perf-Trace-Util/lib/Perf/Trace/Core.py | 121 + .../Perf-Trace-Util/lib/Perf/Trace/EventClass.py | 94 + .../Perf-Trace-Util/lib/Perf/Trace/SchedGui.py | 184 + .../python/Perf-Trace-Util/lib/Perf/Trace/Util.py | 86 + .../python/bin/event_analyzing_sample-record | 8 + .../python/bin/event_analyzing_sample-report | 3 + .../python/bin/failed-syscalls-by-pid-record | 2 + .../python/bin/failed-syscalls-by-pid-report | 10 + .../scripts/python/bin/futex-contention-record | 2 + .../scripts/python/bin/futex-contention-report | 4 + .../perf/scripts/python/bin/net_dropmonitor-record | 2 + .../perf/scripts/python/bin/net_dropmonitor-report | 4 + tools/perf/scripts/python/bin/netdev-times-record | 8 + tools/perf/scripts/python/bin/netdev-times-report | 5 + .../perf/scripts/python/bin/sched-migration-record | 2 + .../perf/scripts/python/bin/sched-migration-report | 3 + tools/perf/scripts/python/bin/sctop-record | 2 + tools/perf/scripts/python/bin/sctop-report | 24 + .../python/bin/syscall-counts-by-pid-record | 2 + .../python/bin/syscall-counts-by-pid-report | 10 + .../perf/scripts/python/bin/syscall-counts-record | 2 + .../perf/scripts/python/bin/syscall-counts-report | 10 + tools/perf/scripts/python/check-perf-trace.py | 82 + .../perf/scripts/python/event_analyzing_sample.py | 189 + .../perf/scripts/python/failed-syscalls-by-pid.py | 73 + tools/perf/scripts/python/futex-contention.py | 50 + tools/perf/scripts/python/net_dropmonitor.py | 75 + tools/perf/scripts/python/netdev-times.py | 464 ++ tools/perf/scripts/python/sched-migration.py | 461 ++ tools/perf/scripts/python/sctop.py | 75 + tools/perf/scripts/python/syscall-counts-by-pid.py | 69 + tools/perf/scripts/python/syscall-counts.py | 59 + tools/perf/tests/attr.c | 183 + tools/perf/tests/attr.py | 332 ++ tools/perf/tests/attr/README | 64 + tools/perf/tests/attr/base-record | 40 + tools/perf/tests/attr/base-stat | 40 + tools/perf/tests/attr/test-record-C0 | 13 + tools/perf/tests/attr/test-record-basic | 5 + tools/perf/tests/attr/test-record-branch-any | 8 + .../perf/tests/attr/test-record-branch-filter-any | 8 + .../tests/attr/test-record-branch-filter-any_call | 8 + .../tests/attr/test-record-branch-filter-any_ret | 8 + tools/perf/tests/attr/test-record-branch-filter-hv | 8 + .../tests/attr/test-record-branch-filter-ind_call | 8 + tools/perf/tests/attr/test-record-branch-filter-k | 8 + tools/perf/tests/attr/test-record-branch-filter-u | 8 + tools/perf/tests/attr/test-record-count | 8 + tools/perf/tests/attr/test-record-data | 8 + tools/perf/tests/attr/test-record-freq | 6 + tools/perf/tests/attr/test-record-graph-default | 6 + tools/perf/tests/attr/test-record-graph-dwarf | 10 + tools/perf/tests/attr/test-record-graph-fp | 6 + tools/perf/tests/attr/test-record-group | 20 + tools/perf/tests/attr/test-record-group1 | 21 + tools/perf/tests/attr/test-record-no-delay | 9 + tools/perf/tests/attr/test-record-no-inherit | 7 + tools/perf/tests/attr/test-record-no-samples | 6 + tools/perf/tests/attr/test-record-period | 7 + tools/perf/tests/attr/test-record-raw | 7 + tools/perf/tests/attr/test-stat-C0 | 9 + tools/perf/tests/attr/test-stat-basic | 6 + tools/perf/tests/attr/test-stat-default | 64 + tools/perf/tests/attr/test-stat-detailed-1 | 101 + tools/perf/tests/attr/test-stat-detailed-2 | 155 + tools/perf/tests/attr/test-stat-detailed-3 | 173 + tools/perf/tests/attr/test-stat-group | 15 + tools/perf/tests/attr/test-stat-group1 | 15 + tools/perf/tests/attr/test-stat-no-inherit | 7 + tools/perf/tests/bp_signal.c | 186 + tools/perf/tests/bp_signal_overflow.c | 126 + tools/perf/tests/builtin-test.c | 219 + tools/perf/tests/dso-data.c | 159 + tools/perf/tests/evsel-roundtrip-name.c | 114 + tools/perf/tests/evsel-tp-sched.c | 84 + tools/perf/tests/hists_link.c | 500 ++ tools/perf/tests/mmap-basic.c | 150 + tools/perf/tests/open-syscall-all-cpus.c | 109 + tools/perf/tests/open-syscall-tp-fields.c | 121 + tools/perf/tests/open-syscall.c | 55 + tools/perf/tests/parse-events.c | 1383 +++++ tools/perf/tests/perf-record.c | 317 ++ tools/perf/tests/pmu.c | 173 + tools/perf/tests/python-use.c | 23 + tools/perf/tests/rdpmc.c | 175 + tools/perf/tests/sw-clock.c | 119 + tools/perf/tests/task-exit.c | 123 + tools/perf/tests/tests.h | 31 + tools/perf/tests/vmlinux-kallsyms.c | 233 + tools/perf/ui/browser.c | 721 +++ tools/perf/ui/browser.h | 74 + tools/perf/ui/browsers/annotate.c | 1009 ++++ tools/perf/ui/browsers/hists.c | 1905 +++++++ tools/perf/ui/browsers/map.c | 130 + tools/perf/ui/browsers/map.h | 6 + tools/perf/ui/browsers/scripts.c | 188 + tools/perf/ui/gtk/annotate.c | 245 + tools/perf/ui/gtk/browser.c | 87 + tools/perf/ui/gtk/gtk.h | 51 + tools/perf/ui/gtk/helpline.c | 57 + tools/perf/ui/gtk/hists.c | 309 ++ tools/perf/ui/gtk/progress.c | 59 + tools/perf/ui/gtk/setup.c | 23 + tools/perf/ui/gtk/util.c | 113 + tools/perf/ui/helpline.c | 73 + tools/perf/ui/helpline.h | 29 + tools/perf/ui/hist.c | 513 ++ tools/perf/ui/keysyms.h | 28 + tools/perf/ui/libslang.h | 29 + tools/perf/ui/progress.c | 26 + tools/perf/ui/progress.h | 18 + tools/perf/ui/setup.c | 53 + tools/perf/ui/stdio/hist.c | 484 ++ tools/perf/ui/tui/helpline.c | 58 + tools/perf/ui/tui/progress.c | 42 + tools/perf/ui/tui/setup.c | 150 + tools/perf/ui/tui/util.c | 243 + tools/perf/ui/ui.h | 39 + tools/perf/ui/util.c | 84 + tools/perf/ui/util.h | 21 + tools/perf/util/PERF-VERSION-GEN | 38 + tools/perf/util/abspath.c | 37 + tools/perf/util/alias.c | 78 + tools/perf/util/annotate.c | 1330 +++++ tools/perf/util/annotate.h | 194 + tools/perf/util/bitmap.c | 31 + tools/perf/util/build-id.c | 104 + tools/perf/util/build-id.h | 19 + tools/perf/util/cache.h | 77 + tools/perf/util/callchain.c | 463 ++ tools/perf/util/callchain.h | 151 + tools/perf/util/cgroup.c | 177 + tools/perf/util/cgroup.h | 17 + tools/perf/util/color.c | 332 ++ tools/perf/util/color.h | 46 + tools/perf/util/config.c | 509 ++ tools/perf/util/cpumap.c | 319 ++ tools/perf/util/cpumap.h | 49 + tools/perf/util/ctype.c | 39 + tools/perf/util/debug.c | 84 + tools/perf/util/debug.h | 20 + tools/perf/util/dso.c | 595 ++ tools/perf/util/dso.h | 148 + tools/perf/util/dwarf-aux.c | 845 +++ tools/perf/util/dwarf-aux.h | 111 + tools/perf/util/environment.c | 9 + tools/perf/util/event.c | 733 +++ tools/perf/util/event.h | 233 + tools/perf/util/evlist.c | 865 +++ tools/perf/util/evlist.h | 167 + tools/perf/util/evsel.c | 1557 ++++++ tools/perf/util/evsel.h | 304 ++ tools/perf/util/exec_cmd.c | 148 + tools/perf/util/exec_cmd.h | 12 + tools/perf/util/generate-cmdlist.sh | 39 + tools/perf/util/header.c | 3158 +++++++++++ tools/perf/util/header.h | 166 + tools/perf/util/help.c | 340 ++ tools/perf/util/help.h | 29 + tools/perf/util/hist.c | 1008 ++++ tools/perf/util/hist.h | 256 + tools/perf/util/hweight.c | 31 + tools/perf/util/include/asm/alternative-asm.h | 8 + tools/perf/util/include/asm/asm-offsets.h | 1 + tools/perf/util/include/asm/bug.h | 22 + tools/perf/util/include/asm/byteorder.h | 2 + tools/perf/util/include/asm/cpufeature.h | 9 + tools/perf/util/include/asm/dwarf2.h | 13 + tools/perf/util/include/asm/hweight.h | 8 + tools/perf/util/include/asm/swab.h | 1 + tools/perf/util/include/asm/system.h | 1 + tools/perf/util/include/asm/uaccess.h | 14 + tools/perf/util/include/asm/unistd_32.h | 1 + tools/perf/util/include/asm/unistd_64.h | 1 + tools/perf/util/include/dwarf-regs.h | 8 + tools/perf/util/include/linux/bitmap.h | 46 + tools/perf/util/include/linux/bitops.h | 158 + tools/perf/util/include/linux/compiler.h | 21 + tools/perf/util/include/linux/const.h | 1 + tools/perf/util/include/linux/ctype.h | 1 + tools/perf/util/include/linux/export.h | 6 + tools/perf/util/include/linux/hash.h | 5 + tools/perf/util/include/linux/kernel.h | 134 + tools/perf/util/include/linux/linkage.h | 13 + tools/perf/util/include/linux/list.h | 29 + tools/perf/util/include/linux/magic.h | 12 + tools/perf/util/include/linux/poison.h | 1 + tools/perf/util/include/linux/prefetch.h | 6 + tools/perf/util/include/linux/rbtree.h | 2 + tools/perf/util/include/linux/rbtree_augmented.h | 2 + tools/perf/util/include/linux/string.h | 3 + tools/perf/util/include/linux/types.h | 29 + tools/perf/util/intlist.c | 131 + tools/perf/util/intlist.h | 75 + tools/perf/util/levenshtein.c | 84 + tools/perf/util/levenshtein.h | 8 + tools/perf/util/machine.c | 1266 +++++ tools/perf/util/machine.h | 159 + tools/perf/util/map.c | 588 ++ tools/perf/util/map.h | 169 + tools/perf/util/pager.c | 100 + tools/perf/util/parse-events.c | 1240 +++++ tools/perf/util/parse-events.h | 112 + tools/perf/util/parse-events.l | 215 + tools/perf/util/parse-events.y | 424 ++ tools/perf/util/parse-options.c | 589 ++ tools/perf/util/parse-options.h | 193 + tools/perf/util/path.c | 157 + tools/perf/util/perf_regs.h | 14 + tools/perf/util/pmu.c | 566 ++ tools/perf/util/pmu.h | 44 + tools/perf/util/pmu.l | 43 + tools/perf/util/pmu.y | 92 + tools/perf/util/probe-event.c | 2374 ++++++++ tools/perf/util/probe-event.h | 140 + tools/perf/util/probe-finder.c | 1588 ++++++ tools/perf/util/probe-finder.h | 107 + tools/perf/util/pstack.c | 75 + tools/perf/util/pstack.h | 14 + tools/perf/util/python-ext-sources | 21 + tools/perf/util/python.c | 1056 ++++ tools/perf/util/quote.c | 54 + tools/perf/util/quote.h | 29 + tools/perf/util/rblist.c | 107 + tools/perf/util/rblist.h | 47 + tools/perf/util/run-command.c | 214 + tools/perf/util/run-command.h | 58 + .../perf/util/scripting-engines/trace-event-perl.c | 643 +++ .../util/scripting-engines/trace-event-python.c | 694 +++ tools/perf/util/session.c | 1567 ++++++ tools/perf/util/session.h | 118 + tools/perf/util/setup.py | 47 + tools/perf/util/sigchain.c | 52 + tools/perf/util/sigchain.h | 10 + tools/perf/util/sort.c | 1005 ++++ tools/perf/util/sort.h | 178 + tools/perf/util/stat.c | 57 + tools/perf/util/stat.h | 16 + tools/perf/util/strbuf.c | 134 + tools/perf/util/strbuf.h | 92 + tools/perf/util/strfilter.c | 199 + tools/perf/util/strfilter.h | 48 + tools/perf/util/string.c | 389 ++ tools/perf/util/strlist.c | 172 + tools/perf/util/strlist.h | 79 + tools/perf/util/svghelper.c | 501 ++ tools/perf/util/svghelper.h | 28 + tools/perf/util/symbol-elf.c | 852 +++ tools/perf/util/symbol-minimal.c | 306 ++ tools/perf/util/symbol.c | 1269 +++++ tools/perf/util/symbol.h | 250 + tools/perf/util/sysfs.c | 60 + tools/perf/util/sysfs.h | 6 + tools/perf/util/target.c | 151 + tools/perf/util/target.h | 65 + tools/perf/util/thread.c | 86 + tools/perf/util/thread.h | 49 + tools/perf/util/thread_map.c | 296 + tools/perf/util/thread_map.h | 29 + tools/perf/util/tool.h | 50 + tools/perf/util/top.c | 106 + tools/perf/util/top.h | 44 + tools/perf/util/trace-event-info.c | 648 +++ tools/perf/util/trace-event-parse.c | 317 ++ tools/perf/util/trace-event-read.c | 432 ++ tools/perf/util/trace-event-scripting.c | 171 + tools/perf/util/trace-event.h | 104 + tools/perf/util/types.h | 24 + tools/perf/util/unwind.c | 571 ++ tools/perf/util/unwind.h | 35 + tools/perf/util/usage.c | 84 + tools/perf/util/util.c | 271 + tools/perf/util/util.h | 279 + tools/perf/util/values.c | 232 + tools/perf/util/values.h | 27 + tools/perf/util/vdso.c | 111 + tools/perf/util/vdso.h | 18 + tools/perf/util/wrapper.c | 41 + tools/perf/util/xyarray.c | 20 + tools/perf/util/xyarray.h | 20 + tools/power/acpi/Makefile | 18 + tools/power/acpi/acpidump.8 | 59 + tools/power/acpi/acpidump.c | 559 ++ tools/power/cpupower/.gitignore | 29 + tools/power/cpupower/Makefile | 302 ++ tools/power/cpupower/README | 49 + tools/power/cpupower/ToDo | 11 + tools/power/cpupower/bench/Makefile | 36 + tools/power/cpupower/bench/README-BENCH | 124 + tools/power/cpupower/bench/benchmark.c | 194 + tools/power/cpupower/bench/benchmark.h | 29 + tools/power/cpupower/bench/config.h | 36 + tools/power/cpupower/bench/cpufreq-bench_plot.sh | 104 + tools/power/cpupower/bench/cpufreq-bench_script.sh | 101 + tools/power/cpupower/bench/example.cfg | 11 + tools/power/cpupower/bench/main.c | 202 + tools/power/cpupower/bench/parse.c | 225 + tools/power/cpupower/bench/parse.h | 53 + tools/power/cpupower/bench/system.c | 191 + tools/power/cpupower/bench/system.h | 29 + tools/power/cpupower/debug/i386/Makefile | 41 + tools/power/cpupower/debug/i386/centrino-decode.c | 113 + tools/power/cpupower/debug/i386/dump_psb.c | 196 + tools/power/cpupower/debug/i386/intel_gsic.c | 78 + .../power/cpupower/debug/i386/powernow-k8-decode.c | 96 + tools/power/cpupower/debug/kernel/Makefile | 23 + .../power/cpupower/debug/kernel/cpufreq-test_tsc.c | 113 + tools/power/cpupower/debug/x86_64/Makefile | 30 + tools/power/cpupower/lib/cpufreq.c | 208 + tools/power/cpupower/lib/cpufreq.h | 223 + tools/power/cpupower/lib/sysfs.c | 672 +++ tools/power/cpupower/lib/sysfs.h | 31 + tools/power/cpupower/man/cpupower-frequency-info.1 | 74 + tools/power/cpupower/man/cpupower-frequency-set.1 | 52 + tools/power/cpupower/man/cpupower-idle-info.1 | 90 + tools/power/cpupower/man/cpupower-info.1 | 19 + tools/power/cpupower/man/cpupower-monitor.1 | 190 + tools/power/cpupower/man/cpupower-set.1 | 94 + tools/power/cpupower/man/cpupower.1 | 72 + tools/power/cpupower/po/cs.po | 944 ++++ tools/power/cpupower/po/de.po | 961 ++++ tools/power/cpupower/po/fr.po | 947 ++++ tools/power/cpupower/po/it.po | 961 ++++ tools/power/cpupower/po/pt.po | 957 ++++ tools/power/cpupower/utils/builtin.h | 11 + tools/power/cpupower/utils/cpufreq-info.c | 668 +++ tools/power/cpupower/utils/cpufreq-set.c | 331 ++ tools/power/cpupower/utils/cpuidle-info.c | 214 + tools/power/cpupower/utils/cpupower-info.c | 135 + tools/power/cpupower/utils/cpupower-set.c | 134 + tools/power/cpupower/utils/cpupower.c | 214 + tools/power/cpupower/utils/helpers/amd.c | 135 + tools/power/cpupower/utils/helpers/bitmask.c | 292 + tools/power/cpupower/utils/helpers/bitmask.h | 33 + tools/power/cpupower/utils/helpers/cpuid.c | 178 + tools/power/cpupower/utils/helpers/helpers.h | 195 + tools/power/cpupower/utils/helpers/misc.c | 27 + tools/power/cpupower/utils/helpers/msr.c | 115 + tools/power/cpupower/utils/helpers/pci.c | 55 + tools/power/cpupower/utils/helpers/sysfs.c | 358 ++ tools/power/cpupower/utils/helpers/sysfs.h | 30 + tools/power/cpupower/utils/helpers/topology.c | 116 + .../cpupower/utils/idle_monitor/amd_fam14h_idle.c | 335 ++ .../cpupower/utils/idle_monitor/cpuidle_sysfs.c | 196 + .../cpupower/utils/idle_monitor/cpupower-monitor.c | 455 ++ .../cpupower/utils/idle_monitor/cpupower-monitor.h | 85 + .../cpupower/utils/idle_monitor/idle_monitors.def | 7 + .../cpupower/utils/idle_monitor/idle_monitors.h | 18 + .../cpupower/utils/idle_monitor/mperf_monitor.c | 338 ++ tools/power/cpupower/utils/idle_monitor/nhm_idle.c | 216 + tools/power/cpupower/utils/idle_monitor/snb_idle.c | 196 + tools/power/cpupower/utils/version-gen.sh | 35 + tools/power/x86/turbostat/Makefile | 22 + tools/power/x86/turbostat/turbostat.8 | 252 + tools/power/x86/turbostat/turbostat.c | 2349 ++++++++ tools/power/x86/x86_energy_perf_policy/Makefile | 10 + .../x86_energy_perf_policy.8 | 104 + .../x86_energy_perf_policy.c | 324 ++ tools/scripts/Makefile.include | 79 + tools/testing/fault-injection/failcmd.sh | 219 + tools/testing/ktest/compare-ktest-sample.pl | 32 + tools/testing/ktest/examples/README | 32 + tools/testing/ktest/examples/crosstests.conf | 260 + tools/testing/ktest/examples/include/bisect.conf | 90 + tools/testing/ktest/examples/include/defaults.conf | 157 + .../testing/ktest/examples/include/min-config.conf | 60 + .../testing/ktest/examples/include/patchcheck.conf | 111 + tools/testing/ktest/examples/include/tests.conf | 74 + tools/testing/ktest/examples/kvm.conf | 88 + tools/testing/ktest/examples/snowball.conf | 53 + tools/testing/ktest/examples/test.conf | 62 + tools/testing/ktest/ktest.pl | 4087 ++++++++++++++ tools/testing/ktest/sample.conf | 1268 +++++ tools/testing/selftests/Makefile | 25 + tools/testing/selftests/README.txt | 42 + tools/testing/selftests/breakpoints/Makefile | 23 + .../selftests/breakpoints/breakpoint_test.c | 394 ++ tools/testing/selftests/cpu-hotplug/Makefile | 6 + tools/testing/selftests/cpu-hotplug/on-off-test.sh | 221 + tools/testing/selftests/efivarfs/Makefile | 12 + tools/testing/selftests/efivarfs/create-read.c | 38 + tools/testing/selftests/efivarfs/efivarfs.sh | 198 + tools/testing/selftests/efivarfs/open-unlink.c | 63 + tools/testing/selftests/ipc/Makefile | 25 + tools/testing/selftests/ipc/msgque.c | 246 + tools/testing/selftests/kcmp/Makefile | 29 + tools/testing/selftests/kcmp/kcmp_test.c | 96 + tools/testing/selftests/memory-hotplug/Makefile | 6 + .../selftests/memory-hotplug/on-off-test.sh | 230 + tools/testing/selftests/mount/Makefile | 17 + .../selftests/mount/unprivileged-remount-test.c | 370 ++ tools/testing/selftests/mqueue/.gitignore | 2 + tools/testing/selftests/mqueue/Makefile | 10 + tools/testing/selftests/mqueue/mq_open_tests.c | 492 ++ tools/testing/selftests/mqueue/mq_perf_tests.c | 741 +++ tools/testing/selftests/net/.gitignore | 3 + tools/testing/selftests/net/Makefile | 19 + tools/testing/selftests/net/psock_fanout.c | 312 ++ tools/testing/selftests/net/psock_lib.h | 127 + tools/testing/selftests/net/psock_tpacket.c | 824 +++ tools/testing/selftests/net/run_afpackettests | 26 + tools/testing/selftests/net/run_netsocktests | 12 + tools/testing/selftests/net/socket.c | 92 + tools/testing/selftests/ptrace/Makefile | 10 + tools/testing/selftests/ptrace/peeksiginfo.c | 214 + tools/testing/selftests/vm/Makefile | 14 + tools/testing/selftests/vm/hugepage-mmap.c | 92 + tools/testing/selftests/vm/hugepage-shm.c | 100 + tools/testing/selftests/vm/map_hugetlb.c | 79 + tools/testing/selftests/vm/run_vmtests | 77 + tools/testing/selftests/vm/thuge-gen.c | 254 + tools/usb/Makefile | 13 + tools/usb/ffs-test.c | 527 ++ tools/usb/hcd-tests.sh | 275 + tools/usb/testusb.c | 537 ++ tools/virtio/Makefile | 14 + tools/virtio/asm/barrier.h | 14 + tools/virtio/linux/bug.h | 10 + tools/virtio/linux/device.h | 2 + tools/virtio/linux/err.h | 26 + tools/virtio/linux/export.h | 5 + tools/virtio/linux/hrtimer.h | 0 tools/virtio/linux/irqreturn.h | 1 + tools/virtio/linux/kernel.h | 112 + tools/virtio/linux/module.h | 1 + tools/virtio/linux/printk.h | 4 + tools/virtio/linux/ratelimit.h | 4 + tools/virtio/linux/scatterlist.h | 189 + tools/virtio/linux/slab.h | 2 + tools/virtio/linux/types.h | 28 + tools/virtio/linux/uaccess.h | 50 + tools/virtio/linux/uio.h | 3 + tools/virtio/linux/virtio.h | 90 + tools/virtio/linux/virtio_config.h | 6 + tools/virtio/linux/virtio_ring.h | 1 + tools/virtio/linux/vringh.h | 1 + tools/virtio/uapi/linux/uio.h | 1 + tools/virtio/uapi/linux/virtio_config.h | 1 + tools/virtio/uapi/linux/virtio_ring.h | 4 + tools/virtio/vhost_test/Makefile | 2 + tools/virtio/vhost_test/vhost_test.c | 1 + tools/virtio/virtio-trace/Makefile | 13 + tools/virtio/virtio-trace/README | 118 + tools/virtio/virtio-trace/trace-agent-ctl.c | 137 + tools/virtio/virtio-trace/trace-agent-rw.c | 192 + tools/virtio/virtio-trace/trace-agent.c | 270 + tools/virtio/virtio-trace/trace-agent.h | 75 + tools/virtio/virtio_test.c | 288 + tools/virtio/vringh_test.c | 741 +++ tools/vm/.gitignore | 2 + tools/vm/Makefile | 22 + tools/vm/page-types.c | 1005 ++++ tools/vm/slabinfo.c | 1393 +++++ 669 files changed, 143164 insertions(+) create mode 100644 tools/Makefile create mode 100644 tools/cgroup/.gitignore create mode 100644 tools/cgroup/Makefile create mode 100644 tools/cgroup/cgroup_event_listener.c create mode 100755 tools/dct/ADC_YuSu.cmp create mode 100755 tools/dct/Application.cmp create mode 100755 tools/dct/DrvGen create mode 100755 tools/dct/DrvGen.exe create mode 100755 tools/dct/EINT_YuSu.cmp create mode 100755 tools/dct/GPIO_YuSu.cmp create mode 100755 tools/dct/GPIO_protect.cmp create mode 100755 tools/dct/I2C_YuSu.cmp create mode 100755 tools/dct/Keypad_YuSu.cmp create mode 100644 tools/dct/MT6574.fig create mode 100755 tools/dct/MT6735.fig create mode 100755 tools/dct/MT6735M.fig create mode 100755 tools/dct/MT6753.fig create mode 100644 tools/dct/MT8173.fig create mode 100755 tools/dct/MT8590.fig create mode 100755 tools/dct/PMIC_MT6322+6333PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6322PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6323+6333PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6323PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6325PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6328PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6331+6332PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6350PMUMP.cmp create mode 100755 tools/dct/PMIC_MT6397PMUMP.cmp create mode 100755 tools/dct/PMIC_NCPMU.cmp create mode 100755 tools/dct/UEM.cmp create mode 100755 tools/dct/Wizard/AST.cmp create mode 100755 tools/dct/Wizard/Bluetooth.cmp create mode 100755 tools/dct/Wizard/CMMB.cmp create mode 100755 tools/dct/Wizard/Camera.cmp create mode 100755 tools/dct/Wizard/FM.cmp create mode 100755 tools/dct/Wizard/GPS.cmp create mode 100755 tools/dct/Wizard/HW_Module.cmp create mode 100755 tools/dct/Wizard/I2C.cmp create mode 100755 tools/dct/Wizard/I2S.cmp create mode 100755 tools/dct/Wizard/Jogball_and_OFN.cmp create mode 100755 tools/dct/Wizard/JogballandOFN_old.cmp create mode 100755 tools/dct/Wizard/Keypad.cmp create mode 100755 tools/dct/Wizard/LCD.cmp create mode 100755 tools/dct/Wizard/MISC.cmp create mode 100755 tools/dct/Wizard/MotionSensor.cmp create mode 100755 tools/dct/Wizard/PWM.cmp create mode 100755 tools/dct/Wizard/SPI.cmp create mode 100755 tools/dct/Wizard/Sensors.cmp create mode 100755 tools/dct/Wizard/TouchPanel.cmp create mode 100755 tools/dct/Wizard/UART.cmp create mode 100755 tools/dct/Wizard/USB.cmp create mode 100755 tools/dct/Wizard/WIFI.cmp create mode 100755 tools/dct/Wizard/connectivity.cmp create mode 100755 tools/dct/Wizard/mATV.cmp create mode 100755 tools/dct/Wizard/mATV_old.cmp create mode 100644 tools/firewire/Makefile create mode 100644 tools/firewire/decode-fcp.c create mode 100644 tools/firewire/list.h create mode 100644 tools/firewire/nosy-dump.c create mode 100644 tools/firewire/nosy-dump.h create mode 100755 tools/hv/hv_get_dhcp_info.sh create mode 100755 tools/hv/hv_get_dns_info.sh create mode 100644 tools/hv/hv_kvp_daemon.c create mode 100755 tools/hv/hv_set_ifconfig.sh create mode 100644 tools/hv/hv_vss_daemon.c create mode 100644 tools/include/tools/be_byteshift.h create mode 100644 tools/include/tools/le_byteshift.h create mode 100644 tools/lguest/.gitignore create mode 100644 tools/lguest/Makefile create mode 100644 tools/lguest/extract create mode 100644 tools/lguest/lguest.c create mode 100644 tools/lguest/lguest.txt create mode 100644 tools/lib/lk/Makefile create mode 100644 tools/lib/lk/debugfs.c create mode 100644 tools/lib/lk/debugfs.h create mode 100644 tools/lib/traceevent/.gitignore create mode 100644 tools/lib/traceevent/Makefile create mode 100644 tools/lib/traceevent/event-parse.c create mode 100644 tools/lib/traceevent/event-parse.h create mode 100644 tools/lib/traceevent/event-utils.h create mode 100644 tools/lib/traceevent/parse-filter.c create mode 100644 tools/lib/traceevent/parse-utils.c create mode 100644 tools/lib/traceevent/trace-seq.c create mode 100644 tools/net/Makefile create mode 100644 tools/net/bpf_jit_disasm.c create mode 100755 tools/nfsd/inject_fault.sh create mode 100644 tools/perf/.gitignore create mode 100644 tools/perf/CREDITS create mode 100644 tools/perf/Documentation/Makefile create mode 100644 tools/perf/Documentation/android.txt create mode 100644 tools/perf/Documentation/asciidoc.conf create mode 100644 tools/perf/Documentation/examples.txt create mode 100644 tools/perf/Documentation/jit-interface.txt create mode 100644 tools/perf/Documentation/manpage-1.72.xsl create mode 100644 tools/perf/Documentation/manpage-base.xsl create mode 100644 tools/perf/Documentation/manpage-bold-literal.xsl create mode 100644 tools/perf/Documentation/manpage-normal.xsl create mode 100644 tools/perf/Documentation/manpage-suppress-sp.xsl create mode 100644 tools/perf/Documentation/perf-annotate.txt create mode 100644 tools/perf/Documentation/perf-archive.txt create mode 100644 tools/perf/Documentation/perf-bench.txt create mode 100644 tools/perf/Documentation/perf-buildid-cache.txt create mode 100644 tools/perf/Documentation/perf-buildid-list.txt create mode 100644 tools/perf/Documentation/perf-diff.txt create mode 100644 tools/perf/Documentation/perf-evlist.txt create mode 100644 tools/perf/Documentation/perf-help.txt create mode 100644 tools/perf/Documentation/perf-inject.txt create mode 100644 tools/perf/Documentation/perf-kmem.txt create mode 100644 tools/perf/Documentation/perf-kvm.txt create mode 100644 tools/perf/Documentation/perf-list.txt create mode 100644 tools/perf/Documentation/perf-lock.txt create mode 100644 tools/perf/Documentation/perf-mem.txt create mode 100644 tools/perf/Documentation/perf-probe.txt create mode 100644 tools/perf/Documentation/perf-record.txt create mode 100644 tools/perf/Documentation/perf-report.txt create mode 100644 tools/perf/Documentation/perf-sched.txt create mode 100644 tools/perf/Documentation/perf-script-perl.txt create mode 100644 tools/perf/Documentation/perf-script-python.txt create mode 100644 tools/perf/Documentation/perf-script.txt create mode 100644 tools/perf/Documentation/perf-stat.txt create mode 100644 tools/perf/Documentation/perf-test.txt create mode 100644 tools/perf/Documentation/perf-timechart.txt create mode 100644 tools/perf/Documentation/perf-top.txt create mode 100644 tools/perf/Documentation/perf-trace.txt create mode 100644 tools/perf/Documentation/perf.txt create mode 100644 tools/perf/Documentation/perfconfig.example create mode 100644 tools/perf/MANIFEST create mode 100644 tools/perf/Makefile create mode 100644 tools/perf/arch/arm/Makefile create mode 100644 tools/perf/arch/arm/util/dwarf-regs.c create mode 100644 tools/perf/arch/common.c create mode 100644 tools/perf/arch/common.h create mode 100644 tools/perf/arch/powerpc/Makefile create mode 100644 tools/perf/arch/powerpc/util/dwarf-regs.c create mode 100644 tools/perf/arch/powerpc/util/header.c create mode 100644 tools/perf/arch/s390/Makefile create mode 100644 tools/perf/arch/s390/util/dwarf-regs.c create mode 100644 tools/perf/arch/sh/Makefile create mode 100644 tools/perf/arch/sh/util/dwarf-regs.c create mode 100644 tools/perf/arch/sparc/Makefile create mode 100644 tools/perf/arch/sparc/util/dwarf-regs.c create mode 100644 tools/perf/arch/x86/Makefile create mode 100644 tools/perf/arch/x86/include/perf_regs.h create mode 100644 tools/perf/arch/x86/util/dwarf-regs.c create mode 100644 tools/perf/arch/x86/util/header.c create mode 100644 tools/perf/arch/x86/util/unwind.c create mode 100644 tools/perf/bash_completion create mode 100644 tools/perf/bench/bench.h create mode 100644 tools/perf/bench/mem-memcpy-arch.h create mode 100644 tools/perf/bench/mem-memcpy-x86-64-asm-def.h create mode 100644 tools/perf/bench/mem-memcpy-x86-64-asm.S create mode 100644 tools/perf/bench/mem-memcpy.c create mode 100644 tools/perf/bench/mem-memset-arch.h create mode 100644 tools/perf/bench/mem-memset-x86-64-asm-def.h create mode 100644 tools/perf/bench/mem-memset-x86-64-asm.S create mode 100644 tools/perf/bench/mem-memset.c create mode 100644 tools/perf/bench/numa.c create mode 100644 tools/perf/bench/sched-messaging.c create mode 100644 tools/perf/bench/sched-pipe.c create mode 100644 tools/perf/builtin-annotate.c create mode 100644 tools/perf/builtin-bench.c create mode 100644 tools/perf/builtin-buildid-cache.c create mode 100644 tools/perf/builtin-buildid-list.c create mode 100644 tools/perf/builtin-diff.c create mode 100644 tools/perf/builtin-evlist.c create mode 100644 tools/perf/builtin-help.c create mode 100644 tools/perf/builtin-inject.c create mode 100644 tools/perf/builtin-kmem.c create mode 100644 tools/perf/builtin-kvm.c create mode 100644 tools/perf/builtin-list.c create mode 100644 tools/perf/builtin-lock.c create mode 100644 tools/perf/builtin-mem.c create mode 100644 tools/perf/builtin-probe.c create mode 100644 tools/perf/builtin-record.c create mode 100644 tools/perf/builtin-report.c create mode 100644 tools/perf/builtin-sched.c create mode 100644 tools/perf/builtin-script.c create mode 100644 tools/perf/builtin-stat.c create mode 100644 tools/perf/builtin-timechart.c create mode 100644 tools/perf/builtin-top.c create mode 100644 tools/perf/builtin-trace.c create mode 100644 tools/perf/builtin.h create mode 100644 tools/perf/command-list.txt create mode 100644 tools/perf/config/feature-tests.mak create mode 100644 tools/perf/config/utilities.mak create mode 100644 tools/perf/design.txt create mode 100644 tools/perf/perf-archive.sh create mode 100644 tools/perf/perf.c create mode 100644 tools/perf/perf.h create mode 100755 tools/perf/python/twatch.py create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/Context.c create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/Context.xs create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/Makefile.PL create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/README create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/lib/Perf/Trace/Context.pm create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/lib/Perf/Trace/Core.pm create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/lib/Perf/Trace/Util.pm create mode 100644 tools/perf/scripts/perl/Perf-Trace-Util/typemap create mode 100644 tools/perf/scripts/perl/bin/check-perf-trace-record create mode 100644 tools/perf/scripts/perl/bin/failed-syscalls-record create mode 100644 tools/perf/scripts/perl/bin/failed-syscalls-report create mode 100644 tools/perf/scripts/perl/bin/rw-by-file-record create mode 100644 tools/perf/scripts/perl/bin/rw-by-file-report create mode 100644 tools/perf/scripts/perl/bin/rw-by-pid-record create mode 100644 tools/perf/scripts/perl/bin/rw-by-pid-report create mode 100644 tools/perf/scripts/perl/bin/rwtop-record create mode 100644 tools/perf/scripts/perl/bin/rwtop-report create mode 100644 tools/perf/scripts/perl/bin/wakeup-latency-record create mode 100644 tools/perf/scripts/perl/bin/wakeup-latency-report create mode 100644 tools/perf/scripts/perl/check-perf-trace.pl create mode 100644 tools/perf/scripts/perl/failed-syscalls.pl create mode 100644 tools/perf/scripts/perl/rw-by-file.pl create mode 100644 tools/perf/scripts/perl/rw-by-pid.pl create mode 100644 tools/perf/scripts/perl/rwtop.pl create mode 100644 tools/perf/scripts/perl/wakeup-latency.pl create mode 100644 tools/perf/scripts/python/Perf-Trace-Util/Context.c create mode 100644 tools/perf/scripts/python/Perf-Trace-Util/lib/Perf/Trace/Core.py create mode 100755 tools/perf/scripts/python/Perf-Trace-Util/lib/Perf/Trace/EventClass.py create mode 100644 tools/perf/scripts/python/Perf-Trace-Util/lib/Perf/Trace/SchedGui.py create mode 100644 tools/perf/scripts/python/Perf-Trace-Util/lib/Perf/Trace/Util.py create mode 100644 tools/perf/scripts/python/bin/event_analyzing_sample-record create mode 100644 tools/perf/scripts/python/bin/event_analyzing_sample-report create mode 100644 tools/perf/scripts/python/bin/failed-syscalls-by-pid-record create mode 100644 tools/perf/scripts/python/bin/failed-syscalls-by-pid-report create mode 100644 tools/perf/scripts/python/bin/futex-contention-record create mode 100644 tools/perf/scripts/python/bin/futex-contention-report create mode 100755 tools/perf/scripts/python/bin/net_dropmonitor-record create mode 100755 tools/perf/scripts/python/bin/net_dropmonitor-report create mode 100644 tools/perf/scripts/python/bin/netdev-times-record create mode 100644 tools/perf/scripts/python/bin/netdev-times-report create mode 100644 tools/perf/scripts/python/bin/sched-migration-record create mode 100644 tools/perf/scripts/python/bin/sched-migration-report create mode 100644 tools/perf/scripts/python/bin/sctop-record create mode 100644 tools/perf/scripts/python/bin/sctop-report create mode 100644 tools/perf/scripts/python/bin/syscall-counts-by-pid-record create mode 100644 tools/perf/scripts/python/bin/syscall-counts-by-pid-report create mode 100644 tools/perf/scripts/python/bin/syscall-counts-record create mode 100644 tools/perf/scripts/python/bin/syscall-counts-report create mode 100644 tools/perf/scripts/python/check-perf-trace.py create mode 100644 tools/perf/scripts/python/event_analyzing_sample.py create mode 100644 tools/perf/scripts/python/failed-syscalls-by-pid.py create mode 100644 tools/perf/scripts/python/futex-contention.py create mode 100755 tools/perf/scripts/python/net_dropmonitor.py create mode 100644 tools/perf/scripts/python/netdev-times.py create mode 100644 tools/perf/scripts/python/sched-migration.py create mode 100644 tools/perf/scripts/python/sctop.py create mode 100644 tools/perf/scripts/python/syscall-counts-by-pid.py create mode 100644 tools/perf/scripts/python/syscall-counts.py create mode 100644 tools/perf/tests/attr.c create mode 100644 tools/perf/tests/attr.py create mode 100644 tools/perf/tests/attr/README create mode 100644 tools/perf/tests/attr/base-record create mode 100644 tools/perf/tests/attr/base-stat create mode 100644 tools/perf/tests/attr/test-record-C0 create mode 100644 tools/perf/tests/attr/test-record-basic create mode 100644 tools/perf/tests/attr/test-record-branch-any create mode 100644 tools/perf/tests/attr/test-record-branch-filter-any create mode 100644 tools/perf/tests/attr/test-record-branch-filter-any_call create mode 100644 tools/perf/tests/attr/test-record-branch-filter-any_ret create mode 100644 tools/perf/tests/attr/test-record-branch-filter-hv create mode 100644 tools/perf/tests/attr/test-record-branch-filter-ind_call create mode 100644 tools/perf/tests/attr/test-record-branch-filter-k create mode 100644 tools/perf/tests/attr/test-record-branch-filter-u create mode 100644 tools/perf/tests/attr/test-record-count create mode 100644 tools/perf/tests/attr/test-record-data create mode 100644 tools/perf/tests/attr/test-record-freq create mode 100644 tools/perf/tests/attr/test-record-graph-default create mode 100644 tools/perf/tests/attr/test-record-graph-dwarf create mode 100644 tools/perf/tests/attr/test-record-graph-fp create mode 100644 tools/perf/tests/attr/test-record-group create mode 100644 tools/perf/tests/attr/test-record-group1 create mode 100644 tools/perf/tests/attr/test-record-no-delay create mode 100644 tools/perf/tests/attr/test-record-no-inherit create mode 100644 tools/perf/tests/attr/test-record-no-samples create mode 100644 tools/perf/tests/attr/test-record-period create mode 100644 tools/perf/tests/attr/test-record-raw create mode 100644 tools/perf/tests/attr/test-stat-C0 create mode 100644 tools/perf/tests/attr/test-stat-basic create mode 100644 tools/perf/tests/attr/test-stat-default create mode 100644 tools/perf/tests/attr/test-stat-detailed-1 create mode 100644 tools/perf/tests/attr/test-stat-detailed-2 create mode 100644 tools/perf/tests/attr/test-stat-detailed-3 create mode 100644 tools/perf/tests/attr/test-stat-group create mode 100644 tools/perf/tests/attr/test-stat-group1 create mode 100644 tools/perf/tests/attr/test-stat-no-inherit create mode 100644 tools/perf/tests/bp_signal.c create mode 100644 tools/perf/tests/bp_signal_overflow.c create mode 100644 tools/perf/tests/builtin-test.c create mode 100644 tools/perf/tests/dso-data.c create mode 100644 tools/perf/tests/evsel-roundtrip-name.c create mode 100644 tools/perf/tests/evsel-tp-sched.c create mode 100644 tools/perf/tests/hists_link.c create mode 100644 tools/perf/tests/mmap-basic.c create mode 100644 tools/perf/tests/open-syscall-all-cpus.c create mode 100644 tools/perf/tests/open-syscall-tp-fields.c create mode 100644 tools/perf/tests/open-syscall.c create mode 100644 tools/perf/tests/parse-events.c create mode 100644 tools/perf/tests/perf-record.c create mode 100644 tools/perf/tests/pmu.c create mode 100644 tools/perf/tests/python-use.c create mode 100644 tools/perf/tests/rdpmc.c create mode 100644 tools/perf/tests/sw-clock.c create mode 100644 tools/perf/tests/task-exit.c create mode 100644 tools/perf/tests/tests.h create mode 100644 tools/perf/tests/vmlinux-kallsyms.c create mode 100644 tools/perf/ui/browser.c create mode 100644 tools/perf/ui/browser.h create mode 100644 tools/perf/ui/browsers/annotate.c create mode 100644 tools/perf/ui/browsers/hists.c create mode 100644 tools/perf/ui/browsers/map.c create mode 100644 tools/perf/ui/browsers/map.h create mode 100644 tools/perf/ui/browsers/scripts.c create mode 100644 tools/perf/ui/gtk/annotate.c create mode 100644 tools/perf/ui/gtk/browser.c create mode 100644 tools/perf/ui/gtk/gtk.h create mode 100644 tools/perf/ui/gtk/helpline.c create mode 100644 tools/perf/ui/gtk/hists.c create mode 100644 tools/perf/ui/gtk/progress.c create mode 100644 tools/perf/ui/gtk/setup.c create mode 100644 tools/perf/ui/gtk/util.c create mode 100644 tools/perf/ui/helpline.c create mode 100644 tools/perf/ui/helpline.h create mode 100644 tools/perf/ui/hist.c create mode 100644 tools/perf/ui/keysyms.h create mode 100644 tools/perf/ui/libslang.h create mode 100644 tools/perf/ui/progress.c create mode 100644 tools/perf/ui/progress.h create mode 100644 tools/perf/ui/setup.c create mode 100644 tools/perf/ui/stdio/hist.c create mode 100644 tools/perf/ui/tui/helpline.c create mode 100644 tools/perf/ui/tui/progress.c create mode 100644 tools/perf/ui/tui/setup.c create mode 100644 tools/perf/ui/tui/util.c create mode 100644 tools/perf/ui/ui.h create mode 100644 tools/perf/ui/util.c create mode 100644 tools/perf/ui/util.h create mode 100755 tools/perf/util/PERF-VERSION-GEN create mode 100644 tools/perf/util/abspath.c create mode 100644 tools/perf/util/alias.c create mode 100644 tools/perf/util/annotate.c create mode 100644 tools/perf/util/annotate.h create mode 100644 tools/perf/util/bitmap.c create mode 100644 tools/perf/util/build-id.c create mode 100644 tools/perf/util/build-id.h create mode 100644 tools/perf/util/cache.h create mode 100644 tools/perf/util/callchain.c create mode 100644 tools/perf/util/callchain.h create mode 100644 tools/perf/util/cgroup.c create mode 100644 tools/perf/util/cgroup.h create mode 100644 tools/perf/util/color.c create mode 100644 tools/perf/util/color.h create mode 100644 tools/perf/util/config.c create mode 100644 tools/perf/util/cpumap.c create mode 100644 tools/perf/util/cpumap.h create mode 100644 tools/perf/util/ctype.c create mode 100644 tools/perf/util/debug.c create mode 100644 tools/perf/util/debug.h create mode 100644 tools/perf/util/dso.c create mode 100644 tools/perf/util/dso.h create mode 100644 tools/perf/util/dwarf-aux.c create mode 100644 tools/perf/util/dwarf-aux.h create mode 100644 tools/perf/util/environment.c create mode 100644 tools/perf/util/event.c create mode 100644 tools/perf/util/event.h create mode 100644 tools/perf/util/evlist.c create mode 100644 tools/perf/util/evlist.h create mode 100644 tools/perf/util/evsel.c create mode 100644 tools/perf/util/evsel.h create mode 100644 tools/perf/util/exec_cmd.c create mode 100644 tools/perf/util/exec_cmd.h create mode 100755 tools/perf/util/generate-cmdlist.sh create mode 100644 tools/perf/util/header.c create mode 100644 tools/perf/util/header.h create mode 100644 tools/perf/util/help.c create mode 100644 tools/perf/util/help.h create mode 100644 tools/perf/util/hist.c create mode 100644 tools/perf/util/hist.h create mode 100644 tools/perf/util/hweight.c create mode 100644 tools/perf/util/include/asm/alternative-asm.h create mode 100644 tools/perf/util/include/asm/asm-offsets.h create mode 100644 tools/perf/util/include/asm/bug.h create mode 100644 tools/perf/util/include/asm/byteorder.h create mode 100644 tools/perf/util/include/asm/cpufeature.h create mode 100644 tools/perf/util/include/asm/dwarf2.h create mode 100644 tools/perf/util/include/asm/hweight.h create mode 100644 tools/perf/util/include/asm/swab.h create mode 100644 tools/perf/util/include/asm/system.h create mode 100644 tools/perf/util/include/asm/uaccess.h create mode 100644 tools/perf/util/include/asm/unistd_32.h create mode 100644 tools/perf/util/include/asm/unistd_64.h create mode 100644 tools/perf/util/include/dwarf-regs.h create mode 100644 tools/perf/util/include/linux/bitmap.h create mode 100644 tools/perf/util/include/linux/bitops.h create mode 100644 tools/perf/util/include/linux/compiler.h create mode 100644 tools/perf/util/include/linux/const.h create mode 100644 tools/perf/util/include/linux/ctype.h create mode 100644 tools/perf/util/include/linux/export.h create mode 100644 tools/perf/util/include/linux/hash.h create mode 100644 tools/perf/util/include/linux/kernel.h create mode 100644 tools/perf/util/include/linux/linkage.h create mode 100644 tools/perf/util/include/linux/list.h create mode 100644 tools/perf/util/include/linux/magic.h create mode 100644 tools/perf/util/include/linux/poison.h create mode 100644 tools/perf/util/include/linux/prefetch.h create mode 100644 tools/perf/util/include/linux/rbtree.h create mode 100644 tools/perf/util/include/linux/rbtree_augmented.h create mode 100644 tools/perf/util/include/linux/string.h create mode 100644 tools/perf/util/include/linux/types.h create mode 100644 tools/perf/util/intlist.c create mode 100644 tools/perf/util/intlist.h create mode 100644 tools/perf/util/levenshtein.c create mode 100644 tools/perf/util/levenshtein.h create mode 100644 tools/perf/util/machine.c create mode 100644 tools/perf/util/machine.h create mode 100644 tools/perf/util/map.c create mode 100644 tools/perf/util/map.h create mode 100644 tools/perf/util/pager.c create mode 100644 tools/perf/util/parse-events.c create mode 100644 tools/perf/util/parse-events.h create mode 100644 tools/perf/util/parse-events.l create mode 100644 tools/perf/util/parse-events.y create mode 100644 tools/perf/util/parse-options.c create mode 100644 tools/perf/util/parse-options.h create mode 100644 tools/perf/util/path.c create mode 100644 tools/perf/util/perf_regs.h create mode 100644 tools/perf/util/pmu.c create mode 100644 tools/perf/util/pmu.h create mode 100644 tools/perf/util/pmu.l create mode 100644 tools/perf/util/pmu.y create mode 100644 tools/perf/util/probe-event.c create mode 100644 tools/perf/util/probe-event.h create mode 100644 tools/perf/util/probe-finder.c create mode 100644 tools/perf/util/probe-finder.h create mode 100644 tools/perf/util/pstack.c create mode 100644 tools/perf/util/pstack.h create mode 100644 tools/perf/util/python-ext-sources create mode 100644 tools/perf/util/python.c create mode 100644 tools/perf/util/quote.c create mode 100644 tools/perf/util/quote.h create mode 100644 tools/perf/util/rblist.c create mode 100644 tools/perf/util/rblist.h create mode 100644 tools/perf/util/run-command.c create mode 100644 tools/perf/util/run-command.h create mode 100644 tools/perf/util/scripting-engines/trace-event-perl.c create mode 100644 tools/perf/util/scripting-engines/trace-event-python.c create mode 100644 tools/perf/util/session.c create mode 100644 tools/perf/util/session.h create mode 100644 tools/perf/util/setup.py create mode 100644 tools/perf/util/sigchain.c create mode 100644 tools/perf/util/sigchain.h create mode 100644 tools/perf/util/sort.c create mode 100644 tools/perf/util/sort.h create mode 100644 tools/perf/util/stat.c create mode 100644 tools/perf/util/stat.h create mode 100644 tools/perf/util/strbuf.c create mode 100644 tools/perf/util/strbuf.h create mode 100644 tools/perf/util/strfilter.c create mode 100644 tools/perf/util/strfilter.h create mode 100644 tools/perf/util/string.c create mode 100644 tools/perf/util/strlist.c create mode 100644 tools/perf/util/strlist.h create mode 100644 tools/perf/util/svghelper.c create mode 100644 tools/perf/util/svghelper.h create mode 100644 tools/perf/util/symbol-elf.c create mode 100644 tools/perf/util/symbol-minimal.c create mode 100644 tools/perf/util/symbol.c create mode 100644 tools/perf/util/symbol.h create mode 100644 tools/perf/util/sysfs.c create mode 100644 tools/perf/util/sysfs.h create mode 100644 tools/perf/util/target.c create mode 100644 tools/perf/util/target.h create mode 100644 tools/perf/util/thread.c create mode 100644 tools/perf/util/thread.h create mode 100644 tools/perf/util/thread_map.c create mode 100644 tools/perf/util/thread_map.h create mode 100644 tools/perf/util/tool.h create mode 100644 tools/perf/util/top.c create mode 100644 tools/perf/util/top.h create mode 100644 tools/perf/util/trace-event-info.c create mode 100644 tools/perf/util/trace-event-parse.c create mode 100644 tools/perf/util/trace-event-read.c create mode 100644 tools/perf/util/trace-event-scripting.c create mode 100644 tools/perf/util/trace-event.h create mode 100644 tools/perf/util/types.h create mode 100644 tools/perf/util/unwind.c create mode 100644 tools/perf/util/unwind.h create mode 100644 tools/perf/util/usage.c create mode 100644 tools/perf/util/util.c create mode 100644 tools/perf/util/util.h create mode 100644 tools/perf/util/values.c create mode 100644 tools/perf/util/values.h create mode 100644 tools/perf/util/vdso.c create mode 100644 tools/perf/util/vdso.h create mode 100644 tools/perf/util/wrapper.c create mode 100644 tools/perf/util/xyarray.c create mode 100644 tools/perf/util/xyarray.h create mode 100644 tools/power/acpi/Makefile create mode 100644 tools/power/acpi/acpidump.8 create mode 100644 tools/power/acpi/acpidump.c create mode 100644 tools/power/cpupower/.gitignore create mode 100644 tools/power/cpupower/Makefile create mode 100644 tools/power/cpupower/README create mode 100644 tools/power/cpupower/ToDo create mode 100644 tools/power/cpupower/bench/Makefile create mode 100644 tools/power/cpupower/bench/README-BENCH create mode 100644 tools/power/cpupower/bench/benchmark.c create mode 100644 tools/power/cpupower/bench/benchmark.h create mode 100644 tools/power/cpupower/bench/config.h create mode 100755 tools/power/cpupower/bench/cpufreq-bench_plot.sh create mode 100755 tools/power/cpupower/bench/cpufreq-bench_script.sh create mode 100644 tools/power/cpupower/bench/example.cfg create mode 100644 tools/power/cpupower/bench/main.c create mode 100644 tools/power/cpupower/bench/parse.c create mode 100644 tools/power/cpupower/bench/parse.h create mode 100644 tools/power/cpupower/bench/system.c create mode 100644 tools/power/cpupower/bench/system.h create mode 100644 tools/power/cpupower/debug/i386/Makefile create mode 100644 tools/power/cpupower/debug/i386/centrino-decode.c create mode 100644 tools/power/cpupower/debug/i386/dump_psb.c create mode 100644 tools/power/cpupower/debug/i386/intel_gsic.c create mode 100644 tools/power/cpupower/debug/i386/powernow-k8-decode.c create mode 100644 tools/power/cpupower/debug/kernel/Makefile create mode 100644 tools/power/cpupower/debug/kernel/cpufreq-test_tsc.c create mode 100644 tools/power/cpupower/debug/x86_64/Makefile create mode 100644 tools/power/cpupower/lib/cpufreq.c create mode 100644 tools/power/cpupower/lib/cpufreq.h create mode 100644 tools/power/cpupower/lib/sysfs.c create mode 100644 tools/power/cpupower/lib/sysfs.h create mode 100644 tools/power/cpupower/man/cpupower-frequency-info.1 create mode 100644 tools/power/cpupower/man/cpupower-frequency-set.1 create mode 100644 tools/power/cpupower/man/cpupower-idle-info.1 create mode 100644 tools/power/cpupower/man/cpupower-info.1 create mode 100644 tools/power/cpupower/man/cpupower-monitor.1 create mode 100644 tools/power/cpupower/man/cpupower-set.1 create mode 100644 tools/power/cpupower/man/cpupower.1 create mode 100644 tools/power/cpupower/po/cs.po create mode 100644 tools/power/cpupower/po/de.po create mode 100644 tools/power/cpupower/po/fr.po create mode 100644 tools/power/cpupower/po/it.po create mode 100644 tools/power/cpupower/po/pt.po create mode 100644 tools/power/cpupower/utils/builtin.h create mode 100644 tools/power/cpupower/utils/cpufreq-info.c create mode 100644 tools/power/cpupower/utils/cpufreq-set.c create mode 100644 tools/power/cpupower/utils/cpuidle-info.c create mode 100644 tools/power/cpupower/utils/cpupower-info.c create mode 100644 tools/power/cpupower/utils/cpupower-set.c create mode 100644 tools/power/cpupower/utils/cpupower.c create mode 100644 tools/power/cpupower/utils/helpers/amd.c create mode 100644 tools/power/cpupower/utils/helpers/bitmask.c create mode 100644 tools/power/cpupower/utils/helpers/bitmask.h create mode 100644 tools/power/cpupower/utils/helpers/cpuid.c create mode 100644 tools/power/cpupower/utils/helpers/helpers.h create mode 100644 tools/power/cpupower/utils/helpers/misc.c create mode 100644 tools/power/cpupower/utils/helpers/msr.c create mode 100644 tools/power/cpupower/utils/helpers/pci.c create mode 100644 tools/power/cpupower/utils/helpers/sysfs.c create mode 100644 tools/power/cpupower/utils/helpers/sysfs.h create mode 100644 tools/power/cpupower/utils/helpers/topology.c create mode 100644 tools/power/cpupower/utils/idle_monitor/amd_fam14h_idle.c create mode 100644 tools/power/cpupower/utils/idle_monitor/cpuidle_sysfs.c create mode 100644 tools/power/cpupower/utils/idle_monitor/cpupower-monitor.c create mode 100644 tools/power/cpupower/utils/idle_monitor/cpupower-monitor.h create mode 100644 tools/power/cpupower/utils/idle_monitor/idle_monitors.def create mode 100644 tools/power/cpupower/utils/idle_monitor/idle_monitors.h create mode 100644 tools/power/cpupower/utils/idle_monitor/mperf_monitor.c create mode 100644 tools/power/cpupower/utils/idle_monitor/nhm_idle.c create mode 100644 tools/power/cpupower/utils/idle_monitor/snb_idle.c create mode 100755 tools/power/cpupower/utils/version-gen.sh create mode 100644 tools/power/x86/turbostat/Makefile create mode 100644 tools/power/x86/turbostat/turbostat.8 create mode 100644 tools/power/x86/turbostat/turbostat.c create mode 100644 tools/power/x86/x86_energy_perf_policy/Makefile create mode 100644 tools/power/x86/x86_energy_perf_policy/x86_energy_perf_policy.8 create mode 100644 tools/power/x86/x86_energy_perf_policy/x86_energy_perf_policy.c create mode 100644 tools/scripts/Makefile.include create mode 100755 tools/testing/fault-injection/failcmd.sh create mode 100755 tools/testing/ktest/compare-ktest-sample.pl create mode 100644 tools/testing/ktest/examples/README create mode 100644 tools/testing/ktest/examples/crosstests.conf create mode 100644 tools/testing/ktest/examples/include/bisect.conf create mode 100644 tools/testing/ktest/examples/include/defaults.conf create mode 100644 tools/testing/ktest/examples/include/min-config.conf create mode 100644 tools/testing/ktest/examples/include/patchcheck.conf create mode 100644 tools/testing/ktest/examples/include/tests.conf create mode 100644 tools/testing/ktest/examples/kvm.conf create mode 100644 tools/testing/ktest/examples/snowball.conf create mode 100644 tools/testing/ktest/examples/test.conf create mode 100755 tools/testing/ktest/ktest.pl create mode 100644 tools/testing/ktest/sample.conf create mode 100644 tools/testing/selftests/Makefile create mode 100644 tools/testing/selftests/README.txt create mode 100644 tools/testing/selftests/breakpoints/Makefile create mode 100644 tools/testing/selftests/breakpoints/breakpoint_test.c create mode 100644 tools/testing/selftests/cpu-hotplug/Makefile create mode 100755 tools/testing/selftests/cpu-hotplug/on-off-test.sh create mode 100644 tools/testing/selftests/efivarfs/Makefile create mode 100644 tools/testing/selftests/efivarfs/create-read.c create mode 100755 tools/testing/selftests/efivarfs/efivarfs.sh create mode 100644 tools/testing/selftests/efivarfs/open-unlink.c create mode 100644 tools/testing/selftests/ipc/Makefile create mode 100644 tools/testing/selftests/ipc/msgque.c create mode 100644 tools/testing/selftests/kcmp/Makefile create mode 100644 tools/testing/selftests/kcmp/kcmp_test.c create mode 100644 tools/testing/selftests/memory-hotplug/Makefile create mode 100755 tools/testing/selftests/memory-hotplug/on-off-test.sh create mode 100644 tools/testing/selftests/mount/Makefile create mode 100644 tools/testing/selftests/mount/unprivileged-remount-test.c create mode 100644 tools/testing/selftests/mqueue/.gitignore create mode 100644 tools/testing/selftests/mqueue/Makefile create mode 100644 tools/testing/selftests/mqueue/mq_open_tests.c create mode 100644 tools/testing/selftests/mqueue/mq_perf_tests.c create mode 100644 tools/testing/selftests/net/.gitignore create mode 100644 tools/testing/selftests/net/Makefile create mode 100644 tools/testing/selftests/net/psock_fanout.c create mode 100644 tools/testing/selftests/net/psock_lib.h create mode 100644 tools/testing/selftests/net/psock_tpacket.c create mode 100644 tools/testing/selftests/net/run_afpackettests create mode 100644 tools/testing/selftests/net/run_netsocktests create mode 100644 tools/testing/selftests/net/socket.c create mode 100644 tools/testing/selftests/ptrace/Makefile create mode 100644 tools/testing/selftests/ptrace/peeksiginfo.c create mode 100644 tools/testing/selftests/vm/Makefile create mode 100644 tools/testing/selftests/vm/hugepage-mmap.c create mode 100644 tools/testing/selftests/vm/hugepage-shm.c create mode 100644 tools/testing/selftests/vm/map_hugetlb.c create mode 100644 tools/testing/selftests/vm/run_vmtests create mode 100644 tools/testing/selftests/vm/thuge-gen.c create mode 100644 tools/usb/Makefile create mode 100644 tools/usb/ffs-test.c create mode 100644 tools/usb/hcd-tests.sh create mode 100644 tools/usb/testusb.c create mode 100644 tools/virtio/Makefile create mode 100644 tools/virtio/asm/barrier.h create mode 100644 tools/virtio/linux/bug.h create mode 100644 tools/virtio/linux/device.h create mode 100644 tools/virtio/linux/err.h create mode 100644 tools/virtio/linux/export.h create mode 100644 tools/virtio/linux/hrtimer.h create mode 100644 tools/virtio/linux/irqreturn.h create mode 100644 tools/virtio/linux/kernel.h create mode 100644 tools/virtio/linux/module.h create mode 100644 tools/virtio/linux/printk.h create mode 100644 tools/virtio/linux/ratelimit.h create mode 100644 tools/virtio/linux/scatterlist.h create mode 100644 tools/virtio/linux/slab.h create mode 100644 tools/virtio/linux/types.h create mode 100644 tools/virtio/linux/uaccess.h create mode 100644 tools/virtio/linux/uio.h create mode 100644 tools/virtio/linux/virtio.h create mode 100644 tools/virtio/linux/virtio_config.h create mode 100644 tools/virtio/linux/virtio_ring.h create mode 100644 tools/virtio/linux/vringh.h create mode 100644 tools/virtio/uapi/linux/uio.h create mode 100644 tools/virtio/uapi/linux/virtio_config.h create mode 100644 tools/virtio/uapi/linux/virtio_ring.h create mode 100644 tools/virtio/vhost_test/Makefile create mode 100644 tools/virtio/vhost_test/vhost_test.c create mode 100644 tools/virtio/virtio-trace/Makefile create mode 100644 tools/virtio/virtio-trace/README create mode 100644 tools/virtio/virtio-trace/trace-agent-ctl.c create mode 100644 tools/virtio/virtio-trace/trace-agent-rw.c create mode 100644 tools/virtio/virtio-trace/trace-agent.c create mode 100644 tools/virtio/virtio-trace/trace-agent.h create mode 100644 tools/virtio/virtio_test.c create mode 100644 tools/virtio/vringh_test.c create mode 100644 tools/vm/.gitignore create mode 100644 tools/vm/Makefile create mode 100644 tools/vm/page-types.c create mode 100644 tools/vm/slabinfo.c (limited to 'tools') diff --git a/tools/Makefile b/tools/Makefile new file mode 100644 index 000000000..41067f304 --- /dev/null +++ b/tools/Makefile @@ -0,0 +1,91 @@ +include scripts/Makefile.include + +help: + @echo 'Possible targets:' + @echo '' + @echo ' cgroup - cgroup tools' + @echo ' cpupower - a tool for all things x86 CPU power' + @echo ' firewire - the userspace part of nosy, an IEEE-1394 traffic sniffer' + @echo ' lguest - a minimal 32-bit x86 hypervisor' + @echo ' perf - Linux performance measurement and analysis tool' + @echo ' selftests - various kernel selftests' + @echo ' turbostat - Intel CPU idle stats and freq reporting tool' + @echo ' usb - USB testing tools' + @echo ' virtio - vhost test module' + @echo ' net - misc networking tools' + @echo ' vm - misc vm tools' + @echo ' x86_energy_perf_policy - Intel energy policy tool' + @echo '' + @echo 'You can do:' + @echo ' $$ make -C tools/ _install' + @echo '' + @echo ' from the kernel command line to build and install one of' + @echo ' the tools above' + @echo '' + @echo ' $$ make tools/install' + @echo '' + @echo ' installs all tools.' + @echo '' + @echo 'Cleaning targets:' + @echo '' + @echo ' all of the above with the "_clean" string appended cleans' + @echo ' the respective build directory.' + @echo ' clean: a summary clean target to clean _all_ folders' + +cpupower: FORCE + $(call descend,power/$@) + +cgroup firewire guest usb virtio vm net: FORCE + $(call descend,$@) + +liblk: FORCE + $(call descend,lib/lk) + +perf: liblk FORCE + $(call descend,$@) + +selftests: FORCE + $(call descend,testing/$@) + +turbostat x86_energy_perf_policy: FORCE + $(call descend,power/x86/$@) + +cpupower_install: + $(call descend,power/$(@:_install=),install) + +cgroup_install firewire_install lguest_install perf_install usb_install virtio_install vm_install net_install: + $(call descend,$(@:_install=),install) + +selftests_install: + $(call descend,testing/$(@:_clean=),install) + +turbostat_install x86_energy_perf_policy_install: + $(call descend,power/x86/$(@:_install=),install) + +install: cgroup_install cpupower_install firewire_install lguest_install \ + perf_install selftests_install turbostat_install usb_install \ + virtio_install vm_install net_install x86_energy_perf_policy_install + +cpupower_clean: + $(call descend,power/cpupower,clean) + +cgroup_clean firewire_clean lguest_clean usb_clean virtio_clean vm_clean net_clean: + $(call descend,$(@:_clean=),clean) + +liblk_clean: + $(call descend,lib/lk,clean) + +perf_clean: liblk_clean + $(call descend,$(@:_clean=),clean) + +selftests_clean: + $(call descend,testing/$(@:_clean=),clean) + +turbostat_clean x86_energy_perf_policy_clean: + $(call descend,power/x86/$(@:_clean=),clean) + +clean: cgroup_clean cpupower_clean firewire_clean lguest_clean perf_clean \ + selftests_clean turbostat_clean usb_clean virtio_clean \ + vm_clean net_clean x86_energy_perf_policy_clean + +.PHONY: FORCE diff --git a/tools/cgroup/.gitignore b/tools/cgroup/.gitignore new file mode 100644 index 000000000..633cd9b87 --- /dev/null +++ b/tools/cgroup/.gitignore @@ -0,0 +1 @@ +cgroup_event_listener diff --git a/tools/cgroup/Makefile b/tools/cgroup/Makefile new file mode 100644 index 000000000..b4286196b --- /dev/null +++ b/tools/cgroup/Makefile @@ -0,0 +1,11 @@ +# Makefile for cgroup tools + +CC = $(CROSS_COMPILE)gcc +CFLAGS = -Wall -Wextra + +all: cgroup_event_listener +%: %.c + $(CC) $(CFLAGS) -o $@ $^ + +clean: + $(RM) cgroup_event_listener diff --git a/tools/cgroup/cgroup_event_listener.c b/tools/cgroup/cgroup_event_listener.c new file mode 100644 index 000000000..4eb550720 --- /dev/null +++ b/tools/cgroup/cgroup_event_listener.c @@ -0,0 +1,82 @@ +/* + * cgroup_event_listener.c - Simple listener of cgroup events + * + * Copyright (C) Kirill A. Shutemov + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +#define USAGE_STR "Usage: cgroup_event_listener " + +int main(int argc, char **argv) +{ + int efd = -1; + int cfd = -1; + int event_control = -1; + char event_control_path[PATH_MAX]; + char line[LINE_MAX]; + int ret; + + if (argc != 3) + errx(1, "%s", USAGE_STR); + + cfd = open(argv[1], O_RDONLY); + if (cfd == -1) + err(1, "Cannot open %s", argv[1]); + + ret = snprintf(event_control_path, PATH_MAX, "%s/cgroup.event_control", + dirname(argv[1])); + if (ret >= PATH_MAX) + errx(1, "Path to cgroup.event_control is too long"); + + event_control = open(event_control_path, O_WRONLY); + if (event_control == -1) + err(1, "Cannot open %s", event_control_path); + + efd = eventfd(0, 0); + if (efd == -1) + err(1, "eventfd() failed"); + + ret = snprintf(line, LINE_MAX, "%d %d %s", efd, cfd, argv[2]); + if (ret >= LINE_MAX) + errx(1, "Arguments string is too long"); + + ret = write(event_control, line, strlen(line) + 1); + if (ret == -1) + err(1, "Cannot write to cgroup.event_control"); + + while (1) { + uint64_t result; + + ret = read(efd, &result, sizeof(result)); + if (ret == -1) { + if (errno == EINTR) + continue; + err(1, "Cannot read from eventfd"); + } + assert(ret == sizeof(result)); + + ret = access(event_control_path, W_OK); + if ((ret == -1) && (errno == ENOENT)) { + puts("The cgroup seems to have removed."); + break; + } + + if (ret == -1) + err(1, "cgroup.event_control is not accessible any more"); + + printf("%s %s: crossed\n", argv[1], argv[2]); + } + + return 0; +} diff --git a/tools/dct/ADC_YuSu.cmp b/tools/dct/ADC_YuSu.cmp new file mode 100755 index 000000000..0605853e9 --- /dev/null +++ b/tools/dct/ADC_YuSu.cmp @@ -0,0 +1,19 @@ +[ADC_variables] +TEMPERATURE +TEMPERATURE1 +REF_CURRENT +BATTERY_VOLTAGE +CHARGER_VOLTAGE +LCM_VOLTAGE +HF_MIC +UTMS +ADC_FDD_RF_PARAMS_DYNAMIC_CUSTOM_CH + + + +[cust_adc.h_HEADER] +#ifndef __CUST_AUXADC_TOOL_H +#define __CUST_AUXADC_TOOL_H + +[cust_adc.h_TAILER] +#endif //_CUST_AUXADC_TOOL_H diff --git a/tools/dct/Application.cmp b/tools/dct/Application.cmp new file mode 100755 index 000000000..c3a245403 --- /dev/null +++ b/tools/dct/Application.cmp @@ -0,0 +1,52 @@ +[APPLICATION] +Camera_A +Camera_A2 +Camera_D +Camera_D2 +G_sensor +M_3G_PA +M_sensor +Cap_touch +Cyroscope +WiFi2V8 +WiFi1V8 +Bluetooth +FM_Radio +BFC +OFN +Vibrator +Memory_Card +MHL_POWER_LDO1 +MHL_POWER_LDO2 +SIM2 +mATV +CMMB +eMMC +Others + +[APPLICATION_V2] +CAP_TOUCH_VDD +CAP_TOUCH_VIO +MAIN_CAMERA_POWER_A +MAIN_CAMERA_POWER_D +MAIN_CAMERA_POWER_IO +MAIN_CAMERA_POWER_AF +MHL_POWER_LDO1 +MHL_POWER_LDO2 +SUB_CAMERA_POWER_A +SUB_CAMERA_POWER_D +SUB_CAMERA_POWER_IO +SUB_CAMERA_POWER_AF +ACCELEROMETER_SENSOR_VDD +ACCELEROMETER_SENSOR_VIO +AMBIENT_LIGHT_SENSOR_VDD +AMBIENT_LIGHT_SENSOR_VIO +PROXIMITY_SENSOR_VDD +PROXIMITY_SENSOR_VIO +MAGNETOMETER_SENSOR_VDD +MAGNETOMETER_SENSOR_VIO +GYROSCOPE_VDD +GYROSCOPE_VIO +BAROMETER_VDD +BAROMETER_VIO +MEMORY_CARD_VDD diff --git a/tools/dct/DrvGen b/tools/dct/DrvGen new file mode 100755 index 000000000..f8efb1f4b Binary files /dev/null and b/tools/dct/DrvGen differ diff --git a/tools/dct/DrvGen.exe b/tools/dct/DrvGen.exe new file mode 100755 index 000000000..0ad7dd179 Binary files /dev/null and b/tools/dct/DrvGen.exe differ diff --git a/tools/dct/EINT_YuSu.cmp b/tools/dct/EINT_YuSu.cmp new file mode 100755 index 000000000..daf114342 --- /dev/null +++ b/tools/dct/EINT_YuSu.cmp @@ -0,0 +1,174 @@ +[EINT_variables] +MT7118_WIMAX +MT5921_WIFI +BT +KPD_PWRKEY +KPD_SLIDE +TOUCH_PANEL +MT6326_PMIC +HEADSET +MHALL +HALL_1 +HALL_2 +HALL_3 +FM_RDS +HALL_4 +ALS +OFN +WIFI +COMBO_BGF +COMBO_ALL +GSE_1 +GSE_2 +MSE +MHL +GYRO +ACCDET +OTG_IDDIG +CMMB +NFC +IRQ_NFC +HEADSET_HOOK +MT6329_PMIC +EINT_HDMI_HPD +DT_EXT_MD_EXP +DT_EXT_MD_WDT +DT_EXT_MD_WK_UP +DT_EXT_MD_WK_UP_USB +DT_EXT_MD_EXP +EVDO_DT_EXT_MDM_RDY +EVDO_DT_EXT_MDM_WAKE_AP +EVDO_DT_EXT_MDM_RST_IND +EVDO_DT_EXT_MDM_PWR_ON +EVDO_DT_EXT_UART_MDM_WAKE_AP +EVDO_DT_EXT_MDM_ACK +EVDO_DT_EXT_MDM_FLOW_CTRL +MT6280_USB_WAKEUP +MT6280_WD +SIM1_HOT_PLUG +SIM2_HOT_PLUG +MSDC1_INS +MSDC2_INS +MT6322_PMIC +MT6323_PMIC +MT6333_PMIC +MT6397_PMIC +CHR_STAT +LTE_SDIO +LTE_WDT +EXT_BUCK_OC +EDP_INTN +DSI_TE_1 +VBUS_DETECT_PIN_EINT +HALL_INT +MUIC_TSU6721 + +[MD1_EINT_variables] +MD1_SIM1_HOT_PLUG_EINT +MD1_SIM2_HOT_PLUG_EINT +MD1_SIM3_HOT_PLUG_EINT +MD1_SIM4_HOT_PLUG_EINT + +[MD2_EINT_variables] +AST_DATA_INTR +MD2_SIM1_HOT_PLUG_EINT +MD2_SIM2_HOT_PLUG_EINT + +[SRC_PIN] +PAD_INT_SIM1 +PAD_INT_SIM2 +PAD_SPI_MO +PAD_SPI_CS +PAD_URXD0 +PAD_URTS0 +PAD_URXD3 +PAD_DPI_D0 +PAD_DPI_D2 + + +[cust_eint.h_HEADER] +#ifndef __CUST_EINTH +#define __CUST_EINTH + +#ifdef __cplusplus +extern "C" { +#endif + +#define CUST_EINT_POLARITY_LOW 0 +#define CUST_EINT_POLARITY_HIGH 1 +#define CUST_EINT_DEBOUNCE_DISABLE 0 +#define CUST_EINT_DEBOUNCE_ENABLE 1 +#define CUST_EINT_EDGE_SENSITIVE 0 +#define CUST_EINT_LEVEL_SENSITIVE 1 + +////////////////////////////////////////////////////////////////////////////// + +[cust_eint2.h_HEADER] +#ifndef __CUST_EINTH +#define __CUST_EINTH + +#ifdef __cplusplus +extern "C" { +#endif + +#define CUST_EINTF_TRIGGER_RISING 1 //High Polarity and Edge Sensitive +#define CUST_EINTF_TRIGGER_FALLING 2 //Low Polarity and Edge Sensitive +#define CUST_EINTF_TRIGGER_HIGH 4 //High Polarity and Level Sensitive +#define CUST_EINTF_TRIGGER_LOW 8 //Low Polarity and Level Sensitive + +#define CUST_EINT_DEBOUNCE_DISABLE 0 +#define CUST_EINT_DEBOUNCE_ENABLE 1 + +////////////////////////////////////////////////////////////////////////////// + +[cust_eint_ext.h_HEADER] +#ifndef __CUST_EINT_EXTH +#define __CUST_EINT_EXTH + +#ifdef __cplusplus +extern "C" { +#endif + +#define CUST_EINT_POLARITY_LOW 0 +#define CUST_EINT_POLARITY_HIGH 1 +#define CUST_EINT_DEBOUNCE_DISABLE 0 +#define CUST_EINT_DEBOUNCE_ENABLE 1 +#define CUST_EINT_EDGE_SENSITIVE 0 +#define CUST_EINT_LEVEL_SENSITIVE 1 +////////////////////////////////////////////////////////////////////////////// + +[cust_eint_ext2.h_HEADER] +#ifndef __CUST_EINTH +#define __CUST_EINTH + +#ifdef __cplusplus +extern "C" { +#endif + +#define CUST_EINTF_TRIGGER_RISING 1 //High Polarity and Edge Sensitive +#define CUST_EINTF_TRIGGER_FALLING 2 //Low Polarity and Edge Sensitive +#define CUST_EINTF_TRIGGER_HIGH 4 //High Polarity and Level Sensitive +#define CUST_EINTF_TRIGGER_LOW 8 //Low Polarity and Level Sensitive + +#define CUST_EINT_DEBOUNCE_DISABLE 0 +#define CUST_EINT_DEBOUNCE_ENABLE 1 + +////////////////////////////////////////////////////////////////////////////// + +[cust_eint.h_TAILER] + +////////////////////////////////////////////////////////////////////////////// +#ifdef __cplusplus +} + +#endif +#endif //_CUST_EINT_H + +[cust_eint_ext.h_TAILER] + +////////////////////////////////////////////////////////////////////////////// +#ifdef __cplusplus +} + +#endif +#endif //_CUST_EINT__EXT_H diff --git a/tools/dct/GPIO_YuSu.cmp b/tools/dct/GPIO_YuSu.cmp new file mode 100755 index 000000000..6bc0e4d04 --- /dev/null +++ b/tools/dct/GPIO_YuSu.cmp @@ -0,0 +1,599 @@ +[GPIO_variables] +GPIO_AST_RST_PIN +GPIO_AST_CS_PIN +GPIO_AST_CLK32K_PIN +GPIO_AST_WAKEUP_PIN +GPIO_AST_INTR_PIN +GPIO_AST_WAKEUP_INTR_PIN +GPIO_AST_AFC_SWITCH_PIN +GPIO_HEADSET_INSERT_PIN +GPIO_HEADSET_REMOTE_BUTTON_PIN +GPIO_EXTERNAL_AMPLIFIER_PIN +GPIO_BT_POWREN_PIN +GPIO_BT_RESET_PIN +GPIO_BT_EINT_PIN +GPIO_BT_CLK_PIN +GPIO_CMMB_EINT_PIN +GPIO_CMMB_LDO_EN_PIN +GPIO_CMMB_RST_PIN +GPIO_I2C0_SCA_PIN +GPIO_I2C0_SDA_PIN +GPIO_I2C1_SCA_PIN +GPIO_I2C1_SDA_PIN +GPIO_I2C2_SCA_PIN +GPIO_I2C2_SDA_PIN +GPIO_I2C3_SCA_PIN +GPIO_I2C3_SDA_PIN +GPIO_I2C4_SCA_PIN +GPIO_I2C4_SDA_PIN +GPIO_NFC_FIRM_PIN +GPIO_NFC_EINT_PIN +GPIO_NFC_VENB_PIN +GPIO_NFC_OSC_EN_PIN +GPIO_IRQ_NFC_PIN +GPIO_NFC_RST_PIN +GPIO_SPI_CS_PIN +GPIO_SPI_SCK_PIN +GPIO_SPI_MISO_PIN +GPIO_SPI_MOSI_PIN +GPIO_SPI2_CS_PIN +GPIO_SPI2_SCK_PIN +GPIO_SPI2_MISO_PIN +GPIO_SPI2_MOSI_PIN +GPIO_PWM_1_PIN +GPIO_PWM_2_PIN +GPIO_PWM_3_PIN +GPIO_PWM_4_PIN +GPIO_PWM_5_PIN +GPIO_PWM_6_PIN +GPIO_PWM_7_PIN +GPIO_PWR_AVAIL_WLC +GPIO_PWR_BUTTON_PIN +GPIO_PMIC_EINT_PIN +GPIO_RFIC0_BSI_CK +GPIO_RFIC0_BSI_CS +GPIO_RFIC0_BSI_D0 +GPIO_RFIC0_BSI_D1 +GPIO_RFIC0_BSI_D2 +GPIO_SDHC_EINT_PIN +GPIO_SDHC_MC2CM_PIN +GPIO_SDHC_MC2DA2_PIN +GPIO_SDHC_MC2DA3_PIN +GPIO_SDHC_MC2WP_PIN +GPIO_SDHC_MC2PWRON_PIN +GPIO_SDHC_MC2INS_PIN +GPIO_TDM_REQ +GPIO_DISP_LSCK_PIN +GPIO_DISP_LSA0_PIN +GPIO_DISP_LSDA_PIN +GPIO_DISP_LSCE_PIN +GPIO_DISP_ID0_PIN +GPIO_DISP_ID1_PIN +GPIO_GPS_PWREN_PIN +GPIO_GPS_SYNC_PIN +GPIO_GPS_EINT_PIN +GPIO_GPS_CLK_PIN +GPIO_GPS_RST_PIN +GPIO_GPS_LNA_PIN +GPIO_UART_URXD0_PIN +GPIO_UART_UTXD0_PIN +GPIO_UART_UCTS0_PIN +GPIO_UART_URTS0_PIN +GPIO_UART_URXD1_PIN +GPIO_UART_UTXD1_PIN +GPIO_UART_UCTS1_PIN +GPIO_UART_URTS1_PIN +GPIO_UART_URXD2_PIN +GPIO_UART_UTXD2_PIN +GPIO_UART_UCTS2_PIN +GPIO_UART_URTS2_PIN +GPIO_UART_URXD3_PIN +GPIO_UART_UTXD3_PIN +GPIO_UART_UCTS3_PIN +GPIO_UART_URTS3_PIN +GPIO_UART_URXD4_PIN +GPIO_UART_UTXD4_PIN +GPIO_UART_UCTS4_PIN +GPIO_UART_URTS4_PIN +GPIO_KPD_KCOL0_PIN +GPIO_KPD_KCOL1_PIN +GPIO_KPD_KCOL2_PIN +GPIO_KPD_KCOL3_PIN +GPIO_KPD_KCOL4_PIN +GPIO_KPD_KCOL5_PIN +GPIO_KPD_KCOL6_PIN +GPIO_KPD_KCOL7_PIN +GPIO_KPD_KROW0_PIN +GPIO_KPD_KROW1_PIN +GPIO_KPD_KROW2_PIN +GPIO_KPD_KROW3_PIN +GPIO_KPD_KROW4_PIN +GPIO_KPD_KROW5_PIN +GPIO_KPD_KROW6_PIN +GPIO_KPD_KROW7_PIN +GPIO_CTP_EINT_PIN +GPIO_CTP_EN_PIN +GPIO_CTP_RST_PIN +GPIO_WIFI_RST_PIN +GPIO_WIFI_CLK_PIN +GPIO_WIFI_EINT_PIN +GPIO_WIFI_PMU_EN_PIN +GPIO_WIFI_LDO_EN_PIN +GPIO_BQ_INT_PIN +GPIO_WIMAX_INT_PIN +GPIO_WIMAX_RST_PIN +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_FLASH_EN_PIN +GPIO_CAMERA_FLASH_MODE_PIN +GPIO_CAMERA_LDO_EN_PIN +GPIO_CAMERA_AF_EN_PIN +GPIO_CAMERA_RCN_PIN +GPIO_CAMERA_RCP_PIN +GPIO_CAMERA_RDN0_PIN +GPIO_CAMERA_RDP0_PIN +GPIO_CAMERA_RDN1_PIN +GPIO_CAMERA_RDP1_PIN +GPIO_CAMERA_RDN2_PIN +GPIO_CAMERA_RDP2_PIN +GPIO_CAMERA_RDN3_PIN +GPIO_CAMERA_RDP3_PIN +GPIO_CAMERA_RCN_A_PIN +GPIO_CAMERA_RCP_A_PIN +GPIO_CAMERA_RDN0_A_PIN +GPIO_CAMERA_RDP0_A_PIN +GPIO_CAMERA_RDN1_A_PIN +GPIO_CAMERA_RDP1_A_PIN +GPIO_CAMERA_RDN2_A_PIN +GPIO_CAMERA_RDP2_A_PIN +GPIO_CAMERA_RDN3_A_PIN +GPIO_CAMERA_RDP3_A_PIN +GPIO_CAMERA_RCN_B_PIN +GPIO_CAMERA_RCP_B_PIN +GPIO_CAMERA_RDN0_B_PIN +GPIO_CAMERA_RDP0_B_PIN +GPIO_CAMERA_RDN1_B_PIN +GPIO_CAMERA_RDP1_B_PIN +GPIO_PMIC_EINT_PIN +GPIO_PCM_DAICLK_PIN +GPIO_PCM_DAIPCMOUT_PIN +GPIO_PCM_DAIPCMIN_PIN +GPIO_PCM_DAISYNC_PIN +GPIO_JBD_INPUT_UP_PIN +GPIO_JBD_INPUT_LEFT_PIN +GPIO_JBD_INPUT_RIGHT_PIN +GPIO_JBD_INPUT_DOWN_PIN +GPIO_JTAG_TMS_PIN +GPIO_JTAG_TCK_PIN +GPIO_JTAG_TDI_PIN +GPIO_JTAG_TDO_PIN +GPIO_JTAG_TRSTN_PIN +GPIO_QWERTYSLIDE_EINT_PIN +GPIO_CAPTOUCH_EINT_PIN +GPIO_HALL_1_PIN +GPIO_HALL_2_PIN +GPIO_HALL_3_PIN +GPIO_HALL_4_PIN +GPIO_OFN_EINT_PIN +GPIO_OFN_DWN_PIN +GPIO_OFN_RST_PIN +GPIO_MHALL_EINT_PIN +GPIO_FM_RDS_PIN +GPIO_FM_CLK_PIN +GPIO_ALS_EINT_PIN +GPIO_MATV_PWR_ENABLE +GPIO_MATV_N_RST +GPIO_I2S_DATA +GPIO_SPEAKER_EN_PIN +GPIO_RECEIVER_EN_PIN +GPIO_SPEAKER_EARPIECE_SWITCH_PIN +GPIO_SWCHARGER_EN_PIN +GPIO_COMBO_PMU_EN_PIN +GPIO_COMBO_PMUV28_EN_PIN +GPIO_COMBO_RST_PIN +GPIO_COMBO_RTCCLK_PIN +GPIO_COMBO_BGF_EINT_PIN +GPIO_COMBO_ALL_EINT_PIN +GPIO_COMBO_6620_LDO_EN_PIN +GPIO_COMBO_I2S_CK_PIN +GPIO_COMBO_I2S_DAT_PIN +GPIO_COMBO_I2S_WS_PIN +GPIO_COMBO_UTXD_PIN +GPIO_COMBO_URXD_PIN +GPIO_I2S0_CK_PIN +GPIO_I2S0_DAT_PIN +GPIO_I2S0_MCLK_PIN +GPIO_I2S0_WS_PIN +GPIO_I2S1_CK_PIN +GPIO_I2S1_DAT_PIN +GPIO_I2S1_MCLK_PIN +GPIO_I2S1_WS_PIN +GPIO_GSE_1_EINT_PIN +GPIO_GSE_2_EINT_PIN +GPIO_MSE_EINT_PIN +GPIO_GYRO_EINT_PIN +GPIO_ACCDET_EINT_PIN +GPIO_OTG_IDDIG_EINT_PIN +GPIO_OTG_DRVVBUS_PIN +GPIO_VBUS_DETECT_PIN +GPIO_USB_DEFAULT_DEVICE_MODE +GPIO_HEADSET_SW_EN_PIN +GPIO_HEADSET_JPOLE_PIN +GPIO_HEADSET_REMOTE_BUTTON_PIN +GPIO_DISP_LRSTB_PIN +GPIO_HDMI_I2S_OUT_CK_PIN +GPIO_HDMI_I2S_OUT_WS_PIN +GPIO_HDMI_I2S_OUT_DAT_PIN +GPIO_HDMI_I2C_SCL +GPIO_HDMI_I2C_SDA +GPIO_HDMI_POWER_CONTROL +GPIO_HDMI_9024_RESET +GPIO_HDMI_EINT_PIN +GPIO_HDMI_LCD_SW_EN +GPIO_HDMI_PWR_1_2V_EN +GPIO_HIFI_VCCA_EN_PIN +GPIO_HIFI_AVCC_EN_PIN +GPIO_HIFI_DVCC_EN_PIN +GPIO_HW_VER_CHECK_PIN +GPIO_MATV_I2S_CK_PIN +GPIO_MATV_I2S_WS_PIN +GPIO_MATV_I2S_DAT_PIN +GPIO_2G_TX_FILTER_MODE_PIN +GPIO_2G_RX_ACTIVE_PIN +GPIO_4G_RX_ACTIVE_PIN +GPIO_4G_TX_FILTER_MODE_PIN +GPIO_52_RST +GPIO_52_TO_2G +GPIO_52_3G_USB_RESUME +GPIO_52_KCOL0 +GPIO_52_REC_SW +GPIO_52_WD +GPIO_52_USB_SW2 +GPIO_52_TO_3G +GPIO_52_PWR_KEY +GPIO_52_USB_SW1 +GPIO_EXT_BUCK_EN_A_PIN +GPIO_EXT_BUCK_EN_B_PIN +GPIO_EXT_BUCK_IC_EN_PIN +GPIO_EXT_BUCK_OC_EINT_PIN +GPIO_EXT_MD_RST +GPIO_EXT_MD_PWR_KEY +GPIO_EXT_MD_DL_KEY +GPIO_EXT_MD_WD +GPIO_EXT_MD_EXP +GPIO_EXT_MD_WK_AP +GPIO_EXT_MD_META +GPIO_EXT_AP_WK_MD +GPIO_EXT_USB_SW1 +GPIO_EXT_USB_SW2 +GPIO_EXT_USB_RESUME +GPIO_EXT_SSW_S1 +GPIO_EXT_SSW_S2 +GPIO_EXT_SSW_EN +GPIO_EXT_MD_DUMP +GPIO_EXT_SPKAMP_EN_PIN +GPIO_6280_USB_SW1 +GPIO_6280_USB_SW2 +GPIO_6280_KCOL0 +GPIO_6280_USB_WAKEUP_EINT +GPIO_6280_WD +GPIO_6280_RST +GPIO_VIA_MDM_RST +GPIO_VIA_MDM_RST_IND +GPIO_VIA_MDM_PWR_EN +GPIO_VIA_MDM_PWR_ON +GPIO_VIA_CP_BOOT_SEL +GPIO_VIA_AP_WAKE_MDM +GPIO_VIA_MDM_RDY +GPIO_VIA_MDM_WAKE_AP +GPIO_VIA_AP_RDY +GPIO_VIA_SDIO_ACK +GPIO_VIA_FLOW_CTRL +GPIO_VIA_ETS_SEL +GPIO_MSDC0_DAT0 +GPIO_MSDC0_DAT1 +GPIO_MSDC0_DAT2 +GPIO_MSDC0_DAT3 +GPIO_MSDC0_DAT4 +GPIO_MSDC0_DAT5 +GPIO_MSDC0_DAT6 +GPIO_MSDC0_DAT7 +GPIO_MSDC0_CMD +GPIO_MSDC0_CLK +GPIO_MSDC0_RSTB +GPIO_MSDC0_DSL +GPIO_MSDC1_DAT0 +GPIO_MSDC1_DAT1 +GPIO_MSDC1_DAT2 +GPIO_MSDC1_DAT3 +GPIO_MSDC1_CMD +GPIO_MSDC1_CLK +GPIO_MSDC1_INSI +GPIO_MSDC1_SDWPI +GPIO_MSDC2_DAT0 +GPIO_MSDC2_DAT1 +GPIO_MSDC2_DAT2 +GPIO_MSDC2_DAT3 +GPIO_MSDC2_CMD +GPIO_MSDC2_CLK +GPIO_MSDC2_INSI +GPIO_MSDC2_SDWPI +GPIO_MSDC3_DAT0 +GPIO_MSDC3_DAT1 +GPIO_MSDC3_DAT2 +GPIO_MSDC3_DAT3 +GPIO_MSDC3_CMD +GPIO_MSDC3_CLK +GPIO_MSDC4_DAT0 +GPIO_MSDC4_DAT1 +GPIO_MSDC4_DAT2 +GPIO_MSDC4_DAT3 +GPIO_MSDC4_DAT4 +GPIO_MSDC4_DAT5 +GPIO_MSDC4_DAT6 +GPIO_MSDC4_DAT7 +GPIO_MSDC4_CMD +GPIO_MSDC4_CLK +GPIO_MSDC4_RSTB +GPIO_SIM1_HOT_PLUG +GPIO_SIM2_HOT_PLUG +GPIO_FSA8049_PIN +GPIO_ANT_SW_PIN +GPIO_EINT_CHG_STAT_PIN +GPIO_MHL_EINT_PIN +GPIO_MHL_I2S_OUT_CK_PIN +GPIO_MHL_I2S_OUT_WS_PIN +GPIO_MHL_I2S_OUT_DAT_PIN +GPIO_MHL_I2C_SCL +GPIO_MHL_I2C_SDA +GPIO_MHL_RST_B_PIN +GPIO_MHL_POWER_CTRL_PIN +GPIO_EXT_DISP_DPI0_PIN +GPIO_TD_HIF_DCX_PIN +GPIO_TD_HIF_WR_PIN +GPIO_TD_HIF_RD_PIN +GPIO_CAMERA_2_CMPDN_PIN +GPIO_CAMERA_2_CMRST_PIN +GPIO_CAMERA_2_CMMCLK_PIN +GPIO_CAMERA_2_CMRST_PIN +GPIO_CAMERA_2_CMPDN_PIN +GPIO_CHR_PSEL_PIN +GPIO_CHR_CE_PIN +GPIO_CHR_SPM_PIN +GPIO_CAMERA_FLASH_EXT1_PIN +GPIO_CAMERA_FLASH_EXT2_PIN +GPIO_GPS_EN_MT3332 +GPIO_GPS_FRAM_SYNC_MT3332 +GPIO_GPS_TXIND_MT3332 +GPIO_GPS_LNA_MT3332 +GPIO_EXTMD_PCM_CLK_PIN +GPIO_EXTMD_PCM_PCMOUT_PIN +GPIO_EXTMD_PCM_PCMIN_PIN +GPIO_EXTMD_PCM_SYNC_PIN +GPIO_MRG_I2S_PCM_CLK_PIN +GPIO_MRG_I2S_PCM_SYNC_PIN +GPIO_MRG_I2S_PCM_RX_PIN +GPIO_MRG_I2S_PCM_TX_PIN +GPIO_DAIBT_PCM_CLK_PIN +GPIO_DAIBT_PCM_WS_PIN +GPIO_DAIBT_PCM_DI_PIN +GPIO_DAIBT_PCM_DO_PIN +GPIO_DAC_I2S_CLK_PIN +GPIO_DAC_I2S_WS_PIN +GPIO_DAC_I2S_DAT_OUT_PIN +GPIO_DAC_I2S_MCLK_PIN +GPIO_ADC_I2S_CLK_PIN +GPIO_ADC_I2S_WS_PIN +GPIO_ADC_I2S_DAT_IN_PIN +GPIO_ADC_I2S_MCLK_PIN +GPIO_I2S_CLK_PIN +GPIO_I2S_WS_PIN +GPIO_I2S_DAT_OUT_PIN +GPIO_I2S_DAT_IN_PIN +GPIO_AUD_CLK_MOSI_PIN +GPIO_AUD_DAT_MOSI_PIN +GPIO_AUD_DAT_MISO_PIN +GPIO_AUD_EXTHP_EN_PIN +GPIO_AUD_EXTHP_GAIN_PIN +GPIO_AUD_EXTDAC_PWREN_PIN +GPIO_AUD_EXTDAC_RST_PIN +GPIO_AUD_EXTPLL_S0_PIN +GPIO_AUD_EXTPLL_S1_PIN +GPIO_AUD_EXTHPBUF_SDB_PIN +GPIO_AUD_EXTHPBUF_HI_Z_PIN +GPIO_AUD_EXTHPBUF_GAIN_PIN +GPIO_SIM_SWITCH_CLK_PIN +GPIO_SIM_SWITCH_DAT_PIN +GPIO_FDD_BAND_SUPPORT_DETECT_1ST_PIN +GPIO_FDD_BAND_SUPPORT_DETECT_2ND_PIN +GPIO_FDD_BAND_SUPPORT_DETECT_3RD_PIN +GPIO_FDD_BAND_SUPPORT_DETECT_4TH_PIN +GPIO_FDD_BAND_SUPPORT_DETECT_5TH_PIN +GPIO_FDD_BAND_SUPPORT_DETECT_6TH_PIN +GPIO_SIM1_SCLK +GPIO_SIM1_SRST +GPIO_SIM1_SIO +GPIO_SIM2_SCLK +GPIO_SIM2_SRST +GPIO_SIM2_SIO +GPIO_SWITCH1_1V8_PIN +GPIO_USB_MHL_SW_SEL1 +GPIO_USB_MHL_SW_SEL2 +GPIO_LCM_LED_EN +GPIO_LCM_MIPI2LVDS_EN +GPIO_LCM_MIPI2LVDS_PWR_EN +GPIO_LCM_LVDS_PWR_EN +GPIO_LCM_LVDS_EN +GPIO_LCM_PWR +GPIO_LCM_PWR_EN +GPIO_LCM_PWR2_EN +GPIO_LCM_RST +GPIO_LCM_STB +GPIO_LCM_BL_EN +GPIO_LCM_LVL_SHIFT_EN +GPIO_LCM_BRIDGE_EN +GPIO_SSW_EN_PIN +GPIO_SSW_CH_SWAP_PIN +GPIO_LCD_BIAS_ENP_PIN +GPIO_LCD_BIAS_ENN_PIN +GPIO_LCD_DRV_EN_PIN +GPIO_LCDBL_EN_PIN +GPIO_LCD_ENN +GPIO_LCD_ENP +GPIO_CMCSK +GPIO_CMDAT0 +GPIO_CMDAT1 +GPIO_CMDAT2 +GPIO_CMDAT3 +GPIO_CMDAT4 +GPIO_CMDAT5 +GPIO_CMDAT6 +GPIO_CMDAT7 +GPIO_CMDAT8 +GPIO_CMDAT9 +GPIO_CMHSYNC +GPIO_CMMCLK +GPIO_CMPCLK +GPIO_CMVSYNC +GPIO_COMPASS_RST_PIN +GPIO_EXT_BUCK_EN_PIN +GPIO_EXT_BUCK_VSEL_PIN +GPIO_LTE_SDIO_EINT_PIN +GPIO_LTE_WDT_EINT_PIN +GPIO_LTE_POWER_PIN +GPIO_LTE_RESET_PIN +GPIO_LTE_WK_MD_PIN +GPIO_SWITCH1_1V8_PIN +GPIO_CAMERA_MAIN_DVDD_ENADBLE_PIN +GPIO_HIFI_LDO1V8_EN_PIN +GPIO_SMARTPA_RST_PIN +GPIO_EARPHONE_DETECT_PIN +GPIO_LCD_ENP_PIN +GPIO_HIFI_LDO3V3_EN_PIN +GPIO_CAMERA_MCLK_EN_PIN +GPIO_LCD_ENN_PIN +GPIO_NXPSPA_I2S_DATAOUT_PIN +GPIO_NXPSPA_I2S_DATAIN_PIN +GPIO_NXPSPA_I2S_LRCK_PIN +GPIO_NXPSPA_I2S_BCK_PIN +GPIO_STROBE_LED_TPS61311_STRB0 +GPIO_STROBE_LED_TPS61311_STRB1 +GPIO_STROBE_LED_TPS61311_TXMASK +GPIO_STROBE_LED_TPS61311_NRESET +GPIO_EDP_EINT_PIN +GPIO_EDP_ENPSR_PIN +GPIO_EDP_SYSRSTN_PIN +GPIO_EDP_STANDBY_PIN +GPIO_DSI_TE_PIN +GPIO_EXT_BUCK_OC_EINT_PIN +GPIO_VOW_CLK_MISO_PIN +GPIO_G2_TXEN_PIN +GPIO_G2_TXD3_PIN +GPIO_G2_TXD2_PIN +GPIO_G2_TXD1_PIN +GPIO_G2_TXD0_PIN +GPIO_G2_TXC_PIN +GPIO_G2_RXC_PIN +GPIO_G2_RXD0_PIN +GPIO_G2_RXD1_PIN +GPIO_G2_RXD2_PIN +GPIO_G2_RXD3_PIN +GPIO_ESW_INT_PIN +GPIO_G2_RXDV_PIN +GPIO_MDC_PIN +GPIO_MDIO_PIN +GPIO_ESW_RST_PIN +GPIO_IRRX_PIN +GPIO_HDMI_RX_SDA_PIN +GPIO_HDMI_RX_SCL_PIN +GPIO_MAIN_CAM_ID_PIN +GPIO_SUB_CAM_ID_PIN +GPIO_LCD_MAKER_ID +GPIO_AUDIO_SEL +GPIO_DSV_EN +GPIO_TOUCH_MAKER_ID +GPIO_BAT_ID +GPIO_BATT_ID_PULLUP +GPIO_SENSOR0_I2C_SDA +GPIO_SENSOR0_I2C_SCL +GPIO_COMMON_I2C_SCL +GPIO_COMMON_I2C_SDA +GPIO_LCD_BL_EN +GPIO_DSV_AVEE_EN +GPIO_DSV_AVDD_EN +GPIO_NFC_MODE +GPIO_HP_AMP_EN +GPIO_SPK_AMP_EN +GPIO_SMARTPA_I2S_WS_PIN +GPIO_SMARTPA_I2S_DOUT_PIN +GPIO_SMARTPA_I2S_DIN_PIN +GPIO_SMARTPA_I2S_BCK_PIN +GPIO_SMARTPA_RST_PIN +GPIO_SMARTPA_EINT_PIN +GPIO_SMARTPA_LDO_EN_PIN +GPIO_EXT_SPKAMP2_EN_PIN +GPIO_RCV_SPK_SWITCH_PIN +GPIO_FLASH_LED_EN +GPIO_TORCH_EN +GPIO_LTE_VSRAM_EXT_POWER_EN_PIN +GPIO_CAMERA_KEY1_PIN +GPIO_CAMERA_KEY2_PIN +GPIO_CODEC_SPI_CLK_PIN +GPIO_CODEC_SPI_MISO_PIN +GPIO_CODEC_SPI_MOSI_PIN +GPIO_CODEC_SPI_CS_PIN +GPIO_DTV_SPI_SCK_PIN +GPIO_DTV_SPI_MISO_PIN +GPIO_DTV_SPI_MOSI_PIN +GPIO_DTV_SPI_CS_PIN +GPIO_LCM_RST2_PIN +GPIO_IRTX_OUT_PIN +GPIO_PWRAP_SPI0_MI_PIN +GPIO_PWRAP_SPI0_MO_PIN +GPIO_PWRAP_SPI0_CK_PIN +GPIO_PWRAP_SPI0_CSN_PIN +GPIO_LCD_ID_PIN +GPIO_HALL_SWITH_EINT_PIN +GPIO_MUIC_EINT_PIN +GPIO_MT8193_BUS_SWITCH_PIN + +[GPIO_MODE] +GPIO = _M_GPIO +CLK = _M_CLK +EINT = _M_EINT +IRQ = _M_IRQ +KROW = _M_KROW +KCOL = _M_KCOL +PWM = _M_PWM + +[GPIO_FREQ] +GPIO_BT_CLK_PIN = CLK_SRC_F32K +GPIO_GPS_CLK_PIN = CLK_SRC_F32K +GPIO_WIFI_CLK_PIN = CLK_SRC_F32K +GPIO_FM_CLK_PIN = CLK_SRC_F32K +GPIO_COMBO_RTCCLK_PIN = CLK_SRC_F32K + +[cust_gpio_boot.h_HEADER] +#ifndef __CUST_GPIO_BOOT_H__ +#define __CUST_GPIO_BOOT_H__ + +[cust_gpio_boot.h_TAILER] +#endif /* __CUST_GPIO_BOOT_H__ */ + + +[cust_gpio_usage.h_HEADER] +#ifndef __CUST_GPIO_USAGE_H__ +#define __CUST_GPIO_USAGE_H__ + +[cust_gpio_usage.h_TAILER] +#endif /* __CUST_GPIO_USAGE_H__ */ + +[cust_gpio_sleep.h_HEADER] +#ifndef __CUST_GPIO_SLEEP_H__ +#define __CUST_GPIO_SLEEP_H__ + +[cust_gpio_sleep.h_TAILER] +#endif /* __CUST_GPIO_SLEEP_H__ */ + diff --git a/tools/dct/GPIO_protect.cmp b/tools/dct/GPIO_protect.cmp new file mode 100755 index 000000000..7f97f2c70 --- /dev/null +++ b/tools/dct/GPIO_protect.cmp @@ -0,0 +1,25 @@ +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 +[PROTECT] +;GPS +PROTECT_0 = gpio_wifi_enable_pin 0 0 1 0 1 0 0 +PROTECT_1 = gpio_wifi_32k_pin 0 0 1 0 1 0 0 +PROTECT_2 = gpio_wifi_ext_rst_pin 0 0 1 0 1 0 1 +;BT +PROTECT_3 = gpio_bt_power_pin 0 0 1 0 1 0 0 +PROTECT_4 = gpio_bt_wakeup_pin 0 0 1 0 1 0 0 +PROTECT_5 = gpio_bt_32k_pin 0 0 1 0 1 0 0 +PROTECT_6 = gpio_bt_utxd3_pin 0 0 1 0 1 0 0 +PROTECT_7 = gpio_bt_urxd3_pin 1 1 0 1 1 0 0 +PROTECT_8 = gpio_bt_ucts3_pin 0 0 1 0 1 0 0 +PROTECT_9 = gpio_bt_urts3_pin 0 0 1 0 1 0 0 +PROTECT_10 = gpio_bt_pcmclk_pin 0 0 1 0 1 0 0 +PROTECT_11 = gpio_bt_pcmsync_pin 0 0 1 0 1 0 0 +PROTECT_12 = gpio_bt_pcmin_pin 1 0 0 1 1 0 0 +PROTECT_13 = gpio_bt_pcmout_pin 0 0 1 0 1 0 0 +PROTECT_14 = gpio_bt_reset_pin 0 0 1 0 1 0 0 +;WIFI +PROTECT_15 = gpio_gps_power_pin 1 0 1 0 1 0 0 +PROTECT_16 = gpio_gps_urxd_pin 0 0 1 0 1 0 0 +PROTECT_17 = gpio_gps_utxd_pin 0 0 1 0 1 0 0 + diff --git a/tools/dct/I2C_YuSu.cmp b/tools/dct/I2C_YuSu.cmp new file mode 100755 index 000000000..bd10a4b6c --- /dev/null +++ b/tools/dct/I2C_YuSu.cmp @@ -0,0 +1,31 @@ +[SLAVE_DEVICE] +CAMERA_MAIN +CAMERA_SUB +CAMERA_MAIN_AF +CAMERA_SUB_AF +EXT_BUCK +EXT_VBAT_BOOST +SWITHING_CHARGER +CAP_TOUCH +MSENSOR +MHL +GSENSOR +PRESSURE +ALSPS +GYRO +TEMPER +STROBE_MAIN +STROBE_MAIN_2 +STROBE_SUB +STROBE_SUB_2 +NFC +EXT_SPEAKER_AMP +EXT_DISP +I2C_LCD_BIAS + +[cust_i2c.h_HEADER] +#ifndef _CUST_I2C_H +#define _CUST_I2C_H + +[cust_i2c.h_TAILER] +#endif /* _CUST_I2C_H */ \ No newline at end of file diff --git a/tools/dct/Keypad_YuSu.cmp b/tools/dct/Keypad_YuSu.cmp new file mode 100755 index 000000000..a1128f63b --- /dev/null +++ b/tools/dct/Keypad_YuSu.cmp @@ -0,0 +1,271 @@ +[Key_definition] +KEY_NONE +KEY_HOME +KEY_BACK +KEY_CALL +KEY_ENDCALL +KEY_VOLUMEDOWN +KEY_VOLUMEUP +KEY_MUTE +KEY_MENU +KEY_UP +KEY_DOWN +KEY_LEFT +KEY_RIGHT +KEY_OK +KEY_FOCUS +KEY_CAMERA +KEY_POWER +KEY_AT +KEY_POUND +KEY_STAR +KEY_DEL +KEY_TAB +KEY_ENTER +KEY_LEFTSHIFT +KEY_COMMA +KEY_DOT +KEY_SLASH +KEY_LEFTALT +KEY_RIGHTALT +KEY_SPACE +KEY_SEARCH +KEY_SYM +KEY_0 +KEY_1 +KEY_2 +KEY_3 +KEY_4 +KEY_5 +KEY_6 +KEY_7 +KEY_8 +KEY_9 +KEY_A +KEY_B +KEY_C +KEY_D +KEY_E +KEY_F +KEY_G +KEY_H +KEY_I +KEY_J +KEY_K +KEY_L +KEY_M +KEY_N +KEY_O +KEY_P +KEY_Q +KEY_R +KEY_S +KEY_T +KEY_U +KEY_V +KEY_W +KEY_X +KEY_Y +KEY_Z + +[Key_code] +0 +228 +158 +231 +107 +212 +51 +111 +52 +108 +107 +42 +528 +102 +105 +42 +139 +113 +352 +228 +116 +106 +100 +217 +53 +57 +227 +127 +15 +103 +114 +115 +11 +2 +3 +4 +5 +6 +7 +8 +9 +10 +30 +48 +46 +32 +18 +33 +34 +35 +23 +36 +37 +38 +50 +49 +24 +25 +16 +19 +31 +20 +22 +47 +17 +45 +21 +44 + + +[Key_code_linux] +0 +102 +158 +231 +107 +114 +115 +113 +139 +103 +108 +105 +106 +352 +528 +212 +116 +215 +228 +227 +111 +15 +28 +42 +51 +52 +53 +56 +100 +57 +217 +127 +11 +2 +3 +4 +5 +6 +7 +8 +9 +10 +30 +48 +46 +32 +18 +33 +34 +35 +23 +36 +37 +38 +50 +49 +24 +25 +16 +19 +31 +20 +22 +47 +17 +45 +21 +44 + +[Power_Key_definition] +KEY_POWER +KEY_ENDCALL + + +[cust_kpd.h_HEADER] +#ifndef _CUST_KPD_H_ +#define _CUST_KPD_H_ +#include +#include + +#define KPD_YES 1 +#define KPD_NO 0 + +/* available keys (Linux keycodes) */ +#define KEY_CALL KEY_SEND +#define KEY_ENDCALL KEY_END +#undef KEY_OK +#define KEY_OK KEY_REPLY /* DPAD_CENTER */ +#define KEY_FOCUS KEY_HP +#define KEY_AT KEY_EMAIL +#define KEY_POUND 228 //KEY_KBDILLUMTOGGLE +#define KEY_STAR 227 //KEY_SWITCHVIDEOMODE +#define KEY_DEL KEY_BACKSPACE +#define KEY_SYM KEY_COMPOSE +/* KEY_HOME */ +/* KEY_BACK */ +/* KEY_VOLUMEDOWN */ +/* KEY_VOLUMEUP */ +/* KEY_MUTE */ +/* KEY_MENU */ +/* KEY_UP */ +/* KEY_DOWN */ +/* KEY_LEFT */ +/* KEY_RIGHT */ +/* KEY_CAMERA */ +/* KEY_POWER */ +/* KEY_TAB */ +/* KEY_ENTER */ +/* KEY_LEFTSHIFT */ +/* KEY_COMMA */ +/* KEY_DOT */ /* PERIOD */ +/* KEY_SLASH */ +/* KEY_LEFTALT */ +/* KEY_RIGHTALT */ +/* KEY_SPACE */ +/* KEY_SEARCH */ +/* KEY_0 ~ KEY_9 */ +/* KEY_A ~ KEY_Z */ + +/* + * Power key's HW keycodes are 8, 17, 26, 35, 44, 53, 62, 71. Only [8] works + * for Power key in Keypad driver, so we set KEY_ENDCALL in [8] because + * EndCall key is Power key in Android. If KPD_PWRKEY_USE_EINT is YES, these + * eight keycodes will not work for Power key. + */ + +[cust_kpd.h_TAILER] +#endif + diff --git a/tools/dct/MT6574.fig b/tools/dct/MT6574.fig new file mode 100644 index 000000000..47fcf54e6 --- /dev/null +++ b/tools/dct/MT6574.fig @@ -0,0 +1,249 @@ +[Chip Type] +Chip = MT6574 +GPIO_Pull_Sel = 1 +PMIC_Config = 1 +PMIC_ON_OFF_CONFIG =1 +EINT_MD1_Config = 1 +POWER_Config = 1 +POWER_COUNT = 3 +GPIO_ModeNum = 8 +AndroidPhone = 1 +SpecialKey_Config = 1 + +[GPIO] +GPIO0 = MODE0(GPIO0) MODE1(PWM1) MODE2(DPI_D4) MODE3() MODE4() MODE5(CONN_DSP_JDO) MODE6(DSPJTD) MODE7() PD +GPIO1 = MODE0(GPIO1) MODE1(PWM2) MODE2(DPI_D5) MODE3(MD_EINT1) MODE4(TDD_TDO) MODE5(CONN_MCU_TDO) MODE6() MODE7() PD +GPIO2 = MODE0(GPIO2) MODE1(CLKM0) MODE2(DPI_D6) MODE3(MD_EINT2) MODE4() MODE5(CONN_MCU_DBGACK_N) MODE6(KCOL4) MODE7() PD +GPIO3 = MODE0(GPIO3) MODE1(CLKM1) MODE2(DPI_D7) MODE3(SPI_MI) MODE4(MD_EINT3) MODE5(CONN_MCU_DBGI_N) MODE6(KCOL5) MODE7() PD +GPIO4 = MODE0(GPIO4) MODE1(CLKM2) MODE2(DPI_D8) MODE3(SPI_MO) MODE4(TDD_TCK) MODE5(CONN_MCU_TCK0) MODE6(CONN_MCU_AICE_JCKC) MODE7() PD +GPIO5 = MODE0(GPIO5) MODE1(UCTS2) MODE2(DPI_D9) MODE3(SPI_CS) MODE4(TDD_TDI) MODE5(CONN_MCU_TDI) MODE6(KCOL6) MODE7() PD +GPIO6 = MODE0(GPIO6) MODE1(URTS2) MODE2(DPI_D10) MODE3(SPI_CK) MODE4(TDD_TRSTN) MODE5(CONN_MCU_TRST_B) MODE6(KCOL7) MODE7() PD +GPIO7 = MODE0(GPIO7) MODE1(UCTS3) MODE2(DPI_D11) MODE3(SDA1) MODE4(TDD_TMS) MODE5(CONN_MCU_TMS) MODE6(CONN_MCU_AICE_JMSC) MODE7() PD +GPIO8 = MODE0(GPIO8) MODE1(URTS3) MODE2(CLKM3) MODE3(SCL1) MODE4(NLD9) MODE5(MD_EINT1) MODE6() MODE7() PD +GPIO9 = MODE0(GPIO9) MODE1(CLKM4) MODE2(SDA2) MODE3(EXT_FRAME_SYNC) MODE4(NWEB) MODE5(MD_EINT2) MODE6() MODE7() PD +GPIO10 = MODE0(GPIO10) MODE1(CLKM5) MODE2(SCL2) MODE3(EXT_FRAME_SYNC) MODE4(NCEB0) MODE5(MD_EINT3) MODE6() MODE7() PD +GPIO11 = MODE0(GPIO11) MODE1(CLKM4) MODE2(PWM2) MODE3(KROW3) MODE4(NLD12) MODE5() MODE6() MODE7() PD +GPIO12 = MODE0(GPIO12) MODE1(CLKM5) MODE2(PWM0) MODE3(KCOL3) MODE4(NLD13) MODE5() MODE6() MODE7(DBG_MON_B[6]) PD +GPIO13 = MODE0(GPIO13) MODE1() MODE2(GPS_FRAME_SYNC) MODE3() MODE4(MD_EINT1) MODE5() MODE6() MODE7(DBG_MON_B[7]) PD +GPIO14 = MODE0(GPIO14) MODE1() MODE2(DAC_DAT_OUT) MODE3(ANT_SEL3) MODE4(MD_EINT2) MODE5(CONN_MCU_DBGACK_N) MODE6() MODE7(DBG_MON_B[8]) PD +GPIO15 = MODE0(GPIO15) MODE1() MODE2(DAC_WS) MODE3(ANT_SEL4) MODE4(MD_EINT3) MODE5(CONN_MCU_DBGI_N) MODE6() MODE7(DBG_MON_B[9]) PD +GPIO16 = MODE0(GPIO16) MODE1() MODE2(DAC_CK) MODE3(ANT_SEL5) MODE4(NLD15) MODE5(CONN_MCU_TRST_B) MODE6() MODE7(DBG_MON_B[10]) PD +GPIO17 = MODE0(GPIO17) MODE1(UCTS0) MODE2(BSI_B_CLK) MODE3(CLKM0) MODE4(IDDIG) MODE5() MODE6() MODE7(DBG_MON_B[11]) PD +GPIO18 = MODE0(GPIO18) MODE1(URTS0) MODE2(BSI_B_DATA0) MODE3(I2SOUT_LRCK) MODE4(DRV_VBUS) MODE5() MODE6() MODE7(DBG_MON_B[12]) PD +GPIO19 = MODE0(GPIO19) MODE1(UCTS1) MODE2(BSI_B_EN) MODE3(I2SOUT_BCK) MODE4(CLKM1) MODE5() MODE6() MODE7(DBG_MON_B[13]) PD +GPIO20 = MODE0(GPIO20) MODE1(URTS1) MODE2(PCM_TX) MODE3(I2SOUT_DATA_OUT) MODE4(CLKM2) MODE5() MODE6() MODE7(DBG_MON_B[14]) PD +GPIO21 = MODE0(GPIO21) MODE1(PWRAP_SPIDO) MODE2(PWRAP_SPIDI) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO22 = MODE0(GPIO22) MODE1(PWRAP_SPIDI) MODE2(PWRAP_SPIDO) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO23 = MODE0(GPIO23) MODE1(PWRAP_SPICK_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO24 = MODE0(GPIO24) MODE1(PWRAP_SPICS_B_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO25 = MODE0(GPIO25) MODE1() MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO26 = MODE0(GPIO26) MODE1(AUD_CLK) MODE2(ADC_CK) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO27 = MODE0(GPIO27) MODE1(AUD_MISO) MODE2(ADC_DAT_IN) MODE3(AUD_MOSI) MODE4() MODE5() MODE6() MODE7() PD +GPIO28 = MODE0(GPIO28) MODE1(AUD_MOSI) MODE2(ADC_WS) MODE3(AUD_MISO) MODE4() MODE5() MODE6() MODE7() PD +GPIO29 = MODE0(GPIO29) MODE1(MD1_SIM1_SCLK) MODE2(MD1_SIM2_SCLK) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO30 = MODE0(GPIO30) MODE1(MD1_SIM1_SRST) MODE2(MD1_SIM2_SRST) MODE3(PWM3) MODE4() MODE5() MODE6() MODE7() PD +GPIO31 = MODE0(GPIO31) MODE1(MD1_SIM1_SDAT) MODE2(MD1_SIM2_SDAT) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO32 = MODE0(GPIO32) MODE1(MD1_SIM2_SCLK) MODE2(MD1_SIM1_SCLK) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO33 = MODE0(GPIO33) MODE1(MD1_SIM2_SRST) MODE2(MD1_SIM1_SRST) MODE3(PWM4) MODE4() MODE5() MODE6() MODE7() PD +GPIO34 = MODE0(GPIO34) MODE1(MD1_SIM2_SDAT) MODE2(MD1_SIM1_SDAT) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B[0]) PU +GPIO35 = MODE0(GPIO35) MODE1(RTC32K_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO36 = MODE0(GPIO36) MODE1(WATCHDOG) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO37 = MODE0(GPIO37) MODE1(SRCLKENA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO38 = MODE0(GPIO38) MODE1(SRCLKENAI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO39 = MODE0(GPIO39) MODE1(URXD2) MODE2(DPI_HSYNC) MODE3(UTXD2) MODE4(MD_URXD) MODE5(SRCCLKENAI2) MODE6(KROW4) MODE7() PD +GPIO40 = MODE0(GPIO40) MODE1(UTXD2) MODE2(DPI_VSYNC) MODE3(URXD2) MODE4(MD_UTXD) MODE5(TDD_TXD) MODE6(KROW5) MODE7() PD +GPIO41 = MODE0(GPIO41) MODE1(URXD3) MODE2(DPI_CK) MODE3(UTXD3) MODE4(UCTS2) MODE5(PWM3) MODE6(KROW6) MODE7() PD +GPIO42 = MODE0(GPIO42) MODE1(UTXD3) MODE2(DPI_DE) MODE3(URXD3) MODE4(URTS2) MODE5(PWM4) MODE6(KROW7) MODE7() PD +GPIO43 = MODE0(GPIO43) MODE1(PCM_CLK0) MODE2(DPI_D0) MODE3(I2SIN1_BCK0) MODE4(I2SOUT_BCK) MODE5(CONN_DSP_JCK) MODE6(DSPJTCK) MODE7() PD +GPIO44 = MODE0(GPIO44) MODE1(PCM_SYNC) MODE2(DPI_D1) MODE3(I2SIN1_LRCK) MODE4(I2SOUT_LRCK) MODE5(CONN_DSP_JINTP) MODE6() MODE7(DBG_MON_B[3]) PD +GPIO45 = MODE0(GPIO45) MODE1(PCM_RX) MODE2(DPI_D2) MODE3(I2SIN1_DATA_IN) MODE4(PCM_TX) MODE5(CONN_DSP_JDI) MODE6() MODE7(DBG_MON_B[4]) PD +GPIO46 = MODE0(GPIO46) MODE1(PCM_TX) MODE2(DPI_D3) MODE3(I2SOUT_DATA_OUT) MODE4(PCM_RX) MODE5(CONN_DSP_JMS) MODE6(DSPJTMS) MODE7(DBG_MON_B[5]) PD +GPIO47 = MODE0(GPIO47) MODE1(ANT_SEL0) MODE2(PWM0) MODE3(CONN_MCU_DBGACK_N) MODE4() MODE5() MODE6() MODE7(DBG_MON_A[0]) PD +GPIO48 = MODE0(GPIO48) MODE1(ANT_SEL1) MODE2(PWM1) MODE3(CONN_MCU_DBGI_N) MODE4() MODE5() MODE6() MODE7(DBG_MON_A[1]) PD +GPIO49 = MODE0(GPIO49) MODE1(ANT_SEL2) MODE2(PWM2) MODE3(CONN_MCU_TRST_B) MODE4() MODE5() MODE6() MODE7(DBG_MON_A[2]) PD +GPIO50 = MODE0(GPIO50) MODE1(BPI_BUS0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[3]) PD +GPIO51 = MODE0(GPIO51) MODE1(BPI_BUS1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[4]) PD +GPIO52 = MODE0(GPIO52) MODE1(BPI_BUS2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[5]) PD +GPIO53 = MODE0(GPIO53) MODE1(BPI_BUS3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[6]) PD +GPIO54 = MODE0(GPIO54) MODE1(BPI_BUS4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[7]) PD +GPIO55 = MODE0(GPIO55) MODE1(BPI_BUS5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[8]) PD +GPIO56 = MODE0(GPIO56) MODE1(BPI_BUS6) MODE2(CLKM0) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[9]) PD +GPIO57 = MODE0(GPIO57) MODE1(BPI_BUS14) MODE2(CLKM1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[10]) PD +GPIO58 = MODE0(GPIO58) MODE1(BPI_BUS15) MODE2(WB_PA_EN) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[11]) PD +GPIO59 = MODE0(GPIO59) MODE1(BPI_BUS7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[12]) PD +GPIO60 = MODE0(GPIO60) MODE1(BPI_BUS8) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[13]) PD +GPIO61 = MODE0(GPIO61) MODE1(BPI_BUS9) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[14]) PD +GPIO62 = MODE0(GPIO62) MODE1(BPI_BUS10) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[15]) PD +GPIO63 = MODE0(GPIO63) MODE1(BPI_BUS11) MODE2(WB_PA_EN) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[16]) PD +GPIO64 = MODE0(GPIO64) MODE1(BPI_BUS12) MODE2(BSI_C_CLK) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[17]) PD +GPIO65 = MODE0(GPIO65) MODE1(BPI_BUS13) MODE2(BSI_C_DATA0) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[18]) PD +GPIO66 = MODE0(GPIO66) MODE1(BSI_A_EN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[19]) PD +GPIO67 = MODE0(GPIO67) MODE1(BSI_A_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[20]) PD +GPIO68 = MODE0(GPIO68) MODE1(BSI_A_DATA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[21]) PD +GPIO69 = MODE0(GPIO69) MODE1(BSI_A_DATA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[22]) PD +GPIO70 = MODE0(GPIO70) MODE1(BSI_A_DATA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[23]) PD +GPIO71 = MODE0(GPIO71) MODE1(TXBPI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[24]) PD +GPIO72 = MODE0(GPIO72) MODE1(VM0) MODE2(CLKM2) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[25]) PD +GPIO73 = MODE0(GPIO73) MODE1(VM1) MODE2(CLKM3) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[26]) PD +GPIO74 = MODE0(GPIO74) MODE1(KROW0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO75 = MODE0(GPIO75) MODE1(KCOL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO76 = MODE0(GPIO76) MODE1(JTMS) MODE2(CONN_MCU_TMS) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO77 = MODE0(GPIO77) MODE1(JTCK) MODE2(CONN_MCU_TCK1) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO78 = MODE0(GPIO78) MODE1(JTDI) MODE2(CONN_MCU_TDI) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO79 = MODE0(GPIO79) MODE1(JTDO) MODE2(CONN_MCU_TDO) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO80 = MODE0(GPIO80) MODE1(SPI_CS) MODE2() MODE3(I2SIN1_DATA_IN) MODE4() MODE5() MODE6() MODE7(DBG_MON_B[15]) PD +GPIO81 = MODE0(GPIO81) MODE1(SPI_CK) MODE2() MODE3(I2SIN1_LRCK) MODE4() MODE5() MODE6() MODE7(DBG_MON_B[16]) PD +GPIO82 = MODE0(GPIO82) MODE1(SPI_MI) MODE2(SPI_MO) MODE3(I2SIN1_BCK1) MODE4() MODE5() MODE6() MODE7(DBG_MON_B[17]) PD +GPIO83 = MODE0(GPIO83) MODE1(SPI_MO) MODE2(SPI_MI) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO84 = MODE0(GPIO84) MODE1(SDA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO85 = MODE0(GPIO85) MODE1(SCL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO86 = MODE0(GPIO86) MODE1(SDA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO87 = MODE0(GPIO87) MODE1(SCL1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO88 = MODE0(GPIO88) MODE1(SDA2) MODE2(PWM1) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO89 = MODE0(GPIO89) MODE1(SCL2) MODE2(PWM2) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO90 = MODE0(GPIO90) MODE1(DISP_PWM) MODE2(PWM1) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO91 = MODE0(GPIO91) MODE1(WB_RSTB) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[27]) PD +GPIO92 = MODE0(GPIO92) MODE1(KROW1) MODE2(IDDIG) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[28]) PD +GPIO93 = MODE0(GPIO93) MODE1(KROW2) MODE2(DRV_VBUS) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[29]) PD +GPIO94 = MODE0(GPIO94) MODE1(F2W_DATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[30]) PD +GPIO95 = MODE0(GPIO95) MODE1(F2W_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[31]) PD +GPIO96 = MODE0(GPIO96) MODE1(WB_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[32]) PD +GPIO97 = MODE0(GPIO97) MODE1(WB_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO98 = MODE0(GPIO98) MODE1(WB_SEN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO99 = MODE0(GPIO99) MODE1(WB_CRTL0) MODE2(DFD_NTRST_XI) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO100 = MODE0(GPIO100) MODE1(WB_CRTL1) MODE2(DFD_TMS_XI) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO101 = MODE0(GPIO101) MODE1(WB_CRTL2) MODE2(DFD_TCK_XI) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO102 = MODE0(GPIO102) MODE1(WB_CRTL3) MODE2(DFD_TDI_XI) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO103 = MODE0(GPIO103) MODE1(WB_CRTL4) MODE2(DFD_TDO) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO104 = MODE0(GPIO104) MODE1(WB_CRTL5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO105 = MODE0(GPIO105) MODE1(I2SIN1_DATA_IN) MODE2(PCM_RX) MODE3(I2SOUT_DATA_OUT) MODE4(DAC_DAT_OUT) MODE5(PWM0) MODE6() MODE7(DBG_MON_B[18]) PD +GPIO106 = MODE0(GPIO106) MODE1(I2SIN1_LRCK) MODE2(PCM_SYNC) MODE3(I2SOUT_LRCK) MODE4(DAC_WS) MODE5(PWM3) MODE6() MODE7(DBG_MON_B[19]) PD +GPIO107 = MODE0(GPIO107) MODE1(I2SIN1_BCK2) MODE2(PCM_CLK1) MODE3(I2SOUT_BCK) MODE4(DAC_CK) MODE5(PWM4) MODE6() MODE7(DBG_MON_B[20]) PD +GPIO108 = MODE0(GPIO108) MODE1(URXD0) MODE2(UTXD0) MODE3(MD_URXD) MODE4() MODE5() MODE6() MODE7(DBG_MON_B[21]) PU +GPIO109 = MODE0(GPIO109) MODE1(UTXD0) MODE2(URXD0) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5() MODE6() MODE7(DBG_MON_B[22]) PU +GPIO110 = MODE0(GPIO110) MODE1(URXD1) MODE2(UTXD1) MODE3(MD_URXD) MODE4() MODE5() MODE6() MODE7(DBG_MON_B[23]) PD +GPIO111 = MODE0(GPIO111) MODE1(UTXD1) MODE2(URXD1) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5() MODE6() MODE7(DBG_MON_B[24]) PD +GPIO112 = MODE0(GPIO112) MODE1(LCM_RST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B[25]) PD +GPIO113 = MODE0(GPIO113) MODE1(DSI_TE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B[26]) PD +GPIO114 = MODE0(GPIO114) MODE1(MSDC2_CMD) MODE2(SDA1) MODE3() MODE4(NLD6) MODE5() MODE6() MODE7(DBG_MON_B[27]) PD +GPIO115 = MODE0(GPIO115) MODE1(MSDC2_CLK) MODE2(SCL1) MODE3() MODE4(NLD8) MODE5() MODE6() MODE7(DBG_MON_B[28]) PD +GPIO116 = MODE0(GPIO116) MODE1(MSDC2_DAT0) MODE2() MODE3() MODE4(NLD0) MODE5(UTXD0) MODE6() MODE7(DBG_MON_B[29]) PD +GPIO117 = MODE0(GPIO117) MODE1(MSDC2_DAT1) MODE2(ANT_SEL3) MODE3(PWM0) MODE4(NLD1) MODE5(URXD0) MODE6() MODE7(DBG_MON_B[30]) PD +GPIO118 = MODE0(GPIO118) MODE1(MSDC2_DAT2) MODE2(ANT_SEL4) MODE3(SDA2) MODE4(NLD2) MODE5(UTXD1) MODE6() MODE7(DBG_MON_B[31]) PD +GPIO119 = MODE0(GPIO119) MODE1(MSDC2_DAT3) MODE2(ANT_SEL5) MODE3(SCL2) MODE4(NLD3) MODE5(URXD1) MODE6() MODE7(DBG_MON_B[32]) PD +GPIO120 = MODE0(GPIO120) MODE1(CMDAT0) MODE2(CMCSD0) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO121 = MODE0(GPIO121) MODE1(CMDAT1) MODE2(CMCSD1) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO122 = MODE0(GPIO122) MODE1(CMMCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO123 = MODE0(GPIO123) MODE1(CMPCLK) MODE2(CMCSK) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO124 = MODE0(GPIO124) MODE1(MSDC1_CMD) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO125 = MODE0(GPIO125) MODE1(MSDC1_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO126 = MODE0(GPIO126) MODE1(MSDC1_DAT0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO127 = MODE0(GPIO127) MODE1(MSDC1_DAT1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO128 = MODE0(GPIO128) MODE1(MSDC1_DAT2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO129 = MODE0(GPIO129) MODE1(MSDC1_DAT3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO130 = MODE0(GPIO130) MODE1(MSDC0_DAT7) MODE2() MODE3() MODE4(NALE) MODE5() MODE6() MODE7() PU +GPIO131 = MODE0(GPIO131) MODE1(MSDC0_DAT6) MODE2() MODE3() MODE4(NCLE) MODE5() MODE6() MODE7() PU +GPIO132 = MODE0(GPIO132) MODE1(MSDC0_DAT5) MODE2() MODE3() MODE4(NLD4) MODE5() MODE6() MODE7() PU +GPIO133 = MODE0(GPIO133) MODE1(MSDC0_DAT4) MODE2() MODE3() MODE4(NLD5) MODE5() MODE6() MODE7() PU +GPIO134 = MODE0(GPIO134) MODE1(MSDC0_RSTB) MODE2() MODE3() MODE4(NLD10) MODE5() MODE6() MODE7() PU +GPIO135 = MODE0(GPIO135) MODE1(MSDC0_CMD) MODE2() MODE3() MODE4(NRNB) MODE5() MODE6() MODE7() PU +GPIO136 = MODE0(GPIO136) MODE1(MSDC0_CLK) MODE2() MODE3() MODE4(NREB) MODE5() MODE6() MODE7() PD +GPIO137 = MODE0(GPIO137) MODE1(MSDC0_DAT3) MODE2() MODE3() MODE4(NLD7) MODE5() MODE6() MODE7(DBG_MON_B[1]) PU +GPIO138 = MODE0(GPIO138) MODE1(MSDC0_DAT2) MODE2() MODE3() MODE4(NLD14) MODE5() MODE6() MODE7() PU +GPIO139 = MODE0(GPIO139) MODE1(MSDC0_DAT1) MODE2() MODE3() MODE4(NLD11) MODE5() MODE6() MODE7(DBG_MON_B[2]) PU +GPIO140 = MODE0(GPIO140) MODE1(MSDC0_DAT0) MODE2() MODE3() MODE4(WATCHDOG) MODE5() MODE6() MODE7() PU +GPIO141 = MODE0(GPIO141) MODE1(RDP0_A) MODE2(CMVSYNC) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO142 = MODE0(GPIO142) MODE1(RDN0_A) MODE2(CMHSYNC) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO143 = MODE0(GPIO143) MODE1(RDP1_A) MODE2(CMDAT3) MODE3(CMCSD3) MODE4() MODE5() MODE6() MODE7() -- +GPIO144 = MODE0(GPIO144) MODE1(RDN1_A) MODE2(CMDAT2) MODE3(CMCSD2) MODE4() MODE5() MODE6() MODE7() -- +GPIO145 = MODE0(GPIO145) MODE1(RCP_A) MODE2(CMDAT7) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO146 = MODE0(GPIO146) MODE1(RCN_A) MODE2(CMDAT6) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO147 = MODE0(GPIO147) MODE1(RDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO148 = MODE0(GPIO148) MODE1(RDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO149 = MODE0(GPIO149) MODE1(RDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO150 = MODE0(GPIO150) MODE1(RDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO151 = MODE0(GPIO151) MODE1(RCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO152 = MODE0(GPIO152) MODE1(RCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO153 = MODE0(GPIO153) MODE1(RDP2) MODE2(CMDAT9) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO154 = MODE0(GPIO154) MODE1(RDN2) MODE2(CMDAT8) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO155 = MODE0(GPIO155) MODE1(RDP3) MODE2(CMDAT5) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO156 = MODE0(GPIO156) MODE1(RDN3) MODE2(CMDAT4) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO157 = MODE0(GPIO157) MODE1(TDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO158 = MODE0(GPIO158) MODE1(TDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO159 = MODE0(GPIO159) MODE1(TDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO160 = MODE0(GPIO160) MODE1(TDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO161 = MODE0(GPIO161) MODE1(TCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO162 = MODE0(GPIO162) MODE1(TCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO163 = MODE0(GPIO163) MODE1(TDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO164 = MODE0(GPIO164) MODE1(TDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO165 = MODE0(GPIO165) MODE1(TDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO166 = MODE0(GPIO166) MODE1(TDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO167 = MODE0(GPIO167) MODE1(KCOL1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO168 = MODE0(GPIO168) MODE1(KCOL2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD + +[GPO] + +[EINT] +EINT_COUNT = 169 +EINT_DEBOUNCE_TIME_COUNT = 12 + + +[EINT_MD1] +EINT_MD1_COUNT = 3 + +[ADC] +ADC_COUNT = 10 + +[ADC_EX_PIN] +0 +1 +2 +3 +4 +5 +12 +13 +14 +15 + + +[POWER] +DVDD28_MSDC1 +DVDD28_MSDC2 +DVDD28_BPI + +[DVDD28_MSDC1] +VMC +VIO18 +VIO28 + +[DVDD28_MSDC2] +VIO18 +VIO28 +VGP1 + +[DVDD28_BPI] +VIO18 +VIO28 + + +[KEYPAD] +KEY_ROW = 8 +KEY_COLUMN = 9 + + +[SPROUT_LICENSE_HEADER] +/* +* Copyright (C) 2011-2014 MediaTek Inc. +* +* This program is free software: you can redistribute it and/or modify it under the terms of the +* GNU General Public License version 2 as published by the Free Software Foundation. +* +* 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. +* +* You should have received a copy of the GNU General Public License along with this program. +* If not, see . +*/ + diff --git a/tools/dct/MT6735.fig b/tools/dct/MT6735.fig new file mode 100755 index 000000000..537754235 --- /dev/null +++ b/tools/dct/MT6735.fig @@ -0,0 +1,333 @@ +[Chip Type] +Chip = MT6735 +GPIO_Pull_Sel = 1 +PMIC_Config = 1 +PMIC_ON_OFF_CONFIG = 1 +GPIO_ModeNum = 8 +EINT_MD1_Config = 1 +EINT_MD1_SRC_PIN = 1 +POWER_Config = 1 +POWER_COUNT = 4 +SpecialKey_Config = 1 +I2C_Config = 1 +CLOCK_BUFFER_CONFIG = 1 +Extend_Key_Config=1 +Reset_Key_Config = 1 +PMIC_APP_Ver = 2 +PMIC_APP_COUNT = 6 +MD1_EINT_SRC_PIN_CFG = 1 +AndroidPhone = 1 + +[GPIO] +GPIO0 = MODE0(GPIO0) MODE1(IDDIG) MODE2(DPI_D4) MODE3(CLKM4) MODE4(EXT_FRAME_SYNC) MODE5(PWM3) MODE6(KCOL2) MODE7(C2K_ARM_EINT0) PD SMT_GROUP(1) +GPIO1 = MODE0(GPIO1) MODE1(PWM2) MODE2(DPI_D5) MODE3(MD_EINT0) MODE4(TDD_TDO) MODE5(CONN_MCU_TDO) MODE6(PTA_RXD) MODE7(C2K_ARM_EINT1) PD SMT_GROUP(1) +GPIO2 = MODE0(GPIO2) MODE1(CLKM0) MODE2(DPI_D6) MODE3(MD_EINT0) MODE4(USB_DRVVBUS) MODE5(CONN_MCU_DBGACK_N) MODE6(PTA_TXD) MODE7(C2K_ARM_EINT2) PD SMT_GROUP(1) +GPIO3 = MODE0(GPIO3) MODE1(CLKM1) MODE2(DPI_D7) MODE3(SPI_MIB) MODE4(MD_EINT0) MODE5(CONN_MCU_DBGI_N) MODE6(CONN_MCU_AICE_TMSC) MODE7(C2K_ARM_EINT3) PD SMT_GROUP(1) +GPIO4 = MODE0(GPIO4) MODE1(CLKM2) MODE2(DPI_D8) MODE3(SPI_MOB) MODE4(TDD_TCK) MODE5(CONN_MCU_TCK[0]) MODE6(CONN_MCU_AICE_TCKC) MODE7(C2K_DM_EINT0) PD SMT_GROUP(1) +GPIO5 = MODE0(GPIO5) MODE1(UCTS2) MODE2(DPI_D9) MODE3(SPI_CSB) MODE4(TDD_TDI) MODE5(CONN_MCU_TDI) MODE6(I2S1_DO) MODE7(MD_URXD) PD SMT_GROUP(2) +GPIO6 = MODE0(GPIO6) MODE1(URTS2) MODE2(DPI_D10) MODE3(SPI_CKB) MODE4(TDD_TRSTN) MODE5(CONN_MCU_TRST_B) MODE6(I2S1_LRCK) MODE7(MD_UTXD) PD SMT_GROUP(2) +GPIO7 = MODE0(GPIO7) MODE1(UCTS3) MODE2(DPI_D11) MODE3(SDA1) MODE4(TDD_TMS) MODE5(CONN_MCU_TMS) MODE6(I2S1_BCK) MODE7(TDD_TXD) PD SMT_GROUP(2) +GPIO8 = MODE0(GPIO8) MODE1(URTS3) MODE2(C2K_UIM0_HOT_PLUG_IN) MODE3(SCL1) MODE4(PCM1_DO1) MODE5(MD_EINT1) MODE6(KCOL4) MODE7(UTXD0) PD SMT_GROUP(2) +GPIO9 = MODE0(GPIO9) MODE1(C2K_UIM1_HOT_PLUG_IN) MODE2(PCM1_DO0) MODE3(I2S3_MCK) MODE4(MD_EINT2) MODE5(CLKM2) MODE6(I2S1_MCK) MODE7(DBG_MON_A29) PD SMT_GROUP(3) +GPIO10 = MODE0(GPIO10) MODE1(PWM1) MODE2(CLKM1) MODE3(KROW2) MODE4(MD_EINT0) MODE5(I2S1_MCK) MODE6(SDA3) MODE7(DBG_MON_A30) PD SMT_GROUP(3) +GPIO11 = MODE0(GPIO11) MODE1(MD_EINT1) MODE2(IRTX_OUT) MODE3(C2K_UIM0_HOT_PLUG_IN) MODE4(CLKM0) MODE5(I2S2_MCK) MODE6(SCL3) MODE7(URXD0) PD SMT_GROUP(3) +GPIO12 = MODE0(GPIO12) MODE1(I2S0_MCK) MODE2(C2K_UIM1_HOT_PLUG_IN) MODE3(KCOL2) MODE4(MD_EINT2) MODE5(IRTX_OUT) MODE6(SRCLKENAI2) MODE7(PCM1_DO1) PD SMT_GROUP(3) +GPIO13 = MODE0(GPIO13) MODE1(WB_CTRL0) MODE2() MODE3(C2K_ARM_EINT0) MODE4() MODE5() MODE6() MODE7(DBG_MON_A0) PD SMT_GROUP(4) +GPIO14 = MODE0(GPIO14) MODE1(WB_CTRL1) MODE2() MODE3(C2K_ARM_EINT1) MODE4() MODE5() MODE6() MODE7(DBG_MON_A1) PD SMT_GROUP(4) +GPIO15 = MODE0(GPIO15) MODE1(WB_CTRL2) MODE2() MODE3(C2K_ARM_EINT2) MODE4() MODE5() MODE6() MODE7(DBG_MON_A2) PD SMT_GROUP(4) +GPIO16 = MODE0(GPIO16) MODE1(WB_CTRL3) MODE2() MODE3(C2K_ARM_EINT3) MODE4() MODE5() MODE6() MODE7(DBG_MON_A3) PD SMT_GROUP(4) +GPIO17 = MODE0(GPIO17) MODE1(WB_CTRL4) MODE2() MODE3(C2K_DM_EINT0) MODE4(WATCHDOG) MODE5() MODE6() MODE7(DBG_MON_A4) PD SMT_GROUP(4) +GPIO18 = MODE0(GPIO18) MODE1(WB_CTRL5) MODE2() MODE3(C2K_DM_EINT1) MODE4() MODE5() MODE6() MODE7(DBG_MON_A5) PD SMT_GROUP(4) +GPIO19 = MODE0(GPIO19) MODE1(ANT_SEL0) MODE2(IRTX_OUT) MODE3(IRDA_TX) MODE4(C2K_UART0_TXD) MODE5(GPS_FRAME_SYNC) MODE6(LTE_UTXD) MODE7(DBG_MON_A6) PD SMT_GROUP(5) +GPIO20 = MODE0(GPIO20) MODE1(ANT_SEL1) MODE2(C2K_UIM1_HOT_PLUG_IN) MODE3(IRDA_RX) MODE4(C2K_UART0_RXD) MODE5(MD_EINT2) MODE6(LTE_URXD) MODE7(DBG_MON_A7) PD SMT_GROUP(5) +GPIO21 = MODE0(GPIO21) MODE1(ANT_SEL2) MODE2(PWM2) MODE3(IRDA_PDN) MODE4(CORESONIC_SWCK) MODE5(MD_EINT1) MODE6(C2K_UIM0_HOT_PLUG_IN) MODE7(DBG_MON_A8) PD SMT_GROUP(5) +GPIO22 = MODE0(GPIO22) MODE1(RDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO23 = MODE0(GPIO23) MODE1(RDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO24 = MODE0(GPIO24) MODE1(RDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO25 = MODE0(GPIO25) MODE1(RDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO26 = MODE0(GPIO26) MODE1(RCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO27 = MODE0(GPIO27) MODE1(RCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO28 = MODE0(GPIO28) MODE1(RDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO29 = MODE0(GPIO29) MODE1(RDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO30 = MODE0(GPIO30) MODE1(RDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO31 = MODE0(GPIO31) MODE1(RDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO32 = MODE0(GPIO32) MODE1(RDN0_A) MODE2(CMHSYNC) MODE3(CMCSD0) MODE4() MODE5() MODE6() MODE7() -- +GPIO33 = MODE0(GPIO33) MODE1(RDP0_A) MODE2(CMVSYNC) MODE3(CMCSD1) MODE4() MODE5() MODE6() MODE7() -- +GPIO34 = MODE0(GPIO34) MODE1(RDN1_A) MODE2(CMDAT9) MODE3(CMCSD2) MODE4() MODE5() MODE6() MODE7() -- +GPIO35 = MODE0(GPIO35) MODE1(RDP1_A) MODE2(CMDAT8) MODE3(CMCSD3) MODE4() MODE5() MODE6() MODE7() -- +GPIO36 = MODE0(GPIO36) MODE1(RCN_A) MODE2(CMDAT7) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO37 = MODE0(GPIO37) MODE1(RCP_A) MODE2(CMDAT6) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO38 = MODE0(GPIO38) MODE1(RDN2_A) MODE2(CMDAT5) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO39 = MODE0(GPIO39) MODE1(RDP2_A) MODE2(CMDAT4) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO40 = MODE0(GPIO40) MODE1(RDN3_A) MODE2(CMDAT3) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO41 = MODE0(GPIO41) MODE1(RDP3_A) MODE2(CMDAT2) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO42 = MODE0(GPIO42) MODE1(CMDAT0) MODE2(CMCSD0) MODE3(CMMCLK1) MODE4() MODE5(ANT_SEL5) MODE6(CLKM5) MODE7(DBG_MON_A9) PD SMT_GROUP(6) +GPIO43 = MODE0(GPIO43) MODE1(CMDAT1) MODE2(CMCSD1) MODE3(CMFLASH) MODE4(MD_EINT0) MODE5(CMMCLK1) MODE6(CLKM4) MODE7(DBG_MON_A10) PD SMT_GROUP(6) +GPIO44 = MODE0(GPIO44) MODE1(CMPCLK) MODE2(CMCSK) MODE3(CMCSD2) MODE4(KCOL3) MODE5(SRCLKENAI2) MODE6(PWM0) MODE7(DBG_MON_A11) PD SMT_GROUP(6) +GPIO45 = MODE0(GPIO45) MODE1(CMMCLK0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A12) PD SMT_GROUP(7) +GPIO46 = MODE0(GPIO46) MODE1(CMMCLK1) MODE2(IDDIG) MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(DM_JTINTP) MODE6(KCOL6) MODE7(DBG_MON_A13) PD SMT_GROUP(7) +GPIO47 = MODE0(GPIO47) MODE1(SDA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(8) +GPIO48 = MODE0(GPIO48) MODE1(SCL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(8) +GPIO49 = MODE0(GPIO49) MODE1(SDA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(9) +GPIO50 = MODE0(GPIO50) MODE1(SCL1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(9) +GPIO51 = MODE0(GPIO51) MODE1(SDA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(10) +GPIO52 = MODE0(GPIO52) MODE1(SCL2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(10) +GPIO53 = MODE0(GPIO53) MODE1(SDA3) MODE2() MODE3(IDDIG) MODE4() MODE5(MD_EINT2) MODE6(C2K_UIM1_HOT_PLUG_IN) MODE7() PU SMT_GROUP(11) +GPIO54 = MODE0(GPIO54) MODE1(SCL3) MODE2() MODE3(IDDIG) MODE4() MODE5(MD_EINT1) MODE6(C2K_UIM0_HOT_PLUG_IN) MODE7() PU SMT_GROUP(11) +GPIO55 = MODE0(GPIO55) MODE1(SRCLKENAI0) MODE2(PWM2) MODE3(CLKM5) MODE4(CORESONIC_SWD) MODE5(ANT_SEL6) MODE6(KROW5) MODE7(DISP_PWM) PD SMT_GROUP(12) +GPIO56 = MODE0(GPIO56) MODE1(SRCLKENA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(13) +GPIO57 = MODE0(GPIO57) MODE1(URXD2) MODE2(DPI_HSYNC0) MODE3(UTXD2) MODE4(MD_URXD) MODE5(SRCLKENAI1) MODE6(KROW4) MODE7(DBG_MON_A14) PD SMT_GROUP(14) +GPIO58 = MODE0(GPIO58) MODE1(UTXD2) MODE2(DPI_VSYNC0) MODE3(URXD2) MODE4(MD_UTXD) MODE5(TDD_TXD) MODE6(KROW5) MODE7(DBG_MON_A15) PD SMT_GROUP(14) +GPIO59 = MODE0(GPIO59) MODE1(URXD3) MODE2(DPI_CK0) MODE3(UTXD3) MODE4(UCTS2) MODE5(PWM3) MODE6(KROW6) MODE7(DBG_MON_A16) PD SMT_GROUP(14) +GPIO60 = MODE0(GPIO60) MODE1(UTXD3) MODE2(DPI_DE0) MODE3(URXD3) MODE4(URTS2) MODE5(PWM4) MODE6(KROW7) MODE7(DBG_MON_A17) PD SMT_GROUP(14) +GPIO61 = MODE0(GPIO61) MODE1(PCM1_CLK[0]) MODE2(DPI_D0) MODE3(I2S0_BCK) MODE4(KROW4) MODE5(ANT_SEL3) MODE6(IRTX_OUT) MODE7(DBG_MON_A18) PD SMT_GROUP(15) +GPIO62 = MODE0(GPIO62) MODE1(PCM1_SYNC) MODE2(DPI_D1) MODE3(I2S0_LRCK) MODE4(KCOL7) MODE5(CLKM3) MODE6(CMFLASH) MODE7(DBG_MON_A19) PD SMT_GROUP(15) +GPIO63 = MODE0(GPIO63) MODE1(PCM1_DI) MODE2(DPI_D2) MODE3(I2S0_DI) MODE4(PCM1_DO0) MODE5(CLKM5) MODE6(KROW3) MODE7(DBG_MON_A20) PD SMT_GROUP(15) +GPIO64 = MODE0(GPIO64) MODE1(PCM1_DO0) MODE2(DPI_D3) MODE3(I2S0_MCK) MODE4(PCM1_DI) MODE5(SRCLKENAI2) MODE6(KCOL5) MODE7(DBG_MON_A21) PD SMT_GROUP(15) +GPIO65 = MODE0(GPIO65) MODE1(SPI_CSA) MODE2(EXT_FRAME_SYNC) MODE3(I2S3_MCK) MODE4(KROW2) MODE5(GPS_FRAME_SYNC) MODE6(PTA_RXD) MODE7(DBG_MON_A22) PD SMT_GROUP(16) +GPIO66 = MODE0(GPIO66) MODE1(SPI_CKA) MODE2(USB_DRVVBUS) MODE3(I2S3_BCK) MODE4(KCOL2) MODE5() MODE6(PTA_TXD) MODE7(DBG_MON_A23) PD SMT_GROUP(16) +GPIO67 = MODE0(GPIO67) MODE1(SPI_MIA) MODE2(SPI_MOA) MODE3(I2S3_DO) MODE4(PTA_RXD) MODE5(IDDIG) MODE6(UCTS1) MODE7(DBG_MON_A24) PD SMT_GROUP(16) +GPIO68 = MODE0(GPIO68) MODE1(SPI_MOA) MODE2(SPI_MIA) MODE3(I2S3_LRCK) MODE4(PTA_TXD) MODE5(ANT_SEL4) MODE6(URTS1) MODE7(DBG_MON_A25) PD SMT_GROUP(16) +GPIO69 = MODE0(GPIO69) MODE1(DISP_PWM) MODE2(PWM1) MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(ANT_SEL7) MODE6(DM_JTINTP) MODE7() PD SMT_GROUP(17) +GPIO70 = MODE0(GPIO70) MODE1(JTMS) MODE2(CONN_MCU_TMS) MODE3(LTE_MD32_JTAG_TMS) MODE4(TDD_TMS) MODE5(CORESONIC_SWD) MODE6(DM_OTMS) MODE7(DFD_TMS) PU SMT_GROUP(18) +GPIO71 = MODE0(GPIO71) MODE1(JTCK) MODE2(CONN_MCU_TCK[1]) MODE3(LTE_MD32_JTAG_TCK) MODE4(TDD_TCK) MODE5(CORESONIC_SWCK) MODE6(DM_OTCK) MODE7(DFD_TCK_XI) PU SMT_GROUP(18) +GPIO72 = MODE0(GPIO72) MODE1(JTDI) MODE2(CONN_MCU_TDI) MODE3(LTE_MD32_JTAG_TDI) MODE4(TDD_TDI) MODE5() MODE6(DM_OTDI) MODE7(DFD_TDI) PU SMT_GROUP(18) +GPIO73 = MODE0(GPIO73) MODE1(JTDO) MODE2(CONN_MCU_TDO) MODE3(LTE_MD32_JTAG_TDO) MODE4(TDD_TDO) MODE5() MODE6(DM_OTDO) MODE7(DFD_TDO) PU SMT_GROUP(18) +GPIO74 = MODE0(GPIO74) MODE1(URXD0) MODE2(UTXD0) MODE3(MD_URXD) MODE4(SDA3) MODE5(C2K_UART0_RXD) MODE6(LTE_URXD) MODE7(AUXIF_ST) PU SMT_GROUP(19) +GPIO75 = MODE0(GPIO75) MODE1(UTXD0) MODE2(URXD0) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5(C2K_UART0_TXD) MODE6(LTE_UTXD) MODE7() PU SMT_GROUP(19) +GPIO76 = MODE0(GPIO76) MODE1(URXD1) MODE2(UTXD1) MODE3(MD_URXD) MODE4(SCL3) MODE5(LTE_URXD) MODE6(C2K_UART0_RXD) MODE7(AUXIF_CLK) PD SMT_GROUP(19) +GPIO77 = MODE0(GPIO77) MODE1(UTXD1) MODE2(URXD1) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5(LTE_UTXD) MODE6(C2K_UART0_TXD) MODE7() PD SMT_GROUP(19) +GPIO78 = MODE0(GPIO78) MODE1(I2S0_DI) MODE2(PCM1_DI) MODE3(I2S3_DO) MODE4(I2S1_DO) MODE5(PWM0) MODE6(I2S2_DI) MODE7(DBG_MON_A26) PD SMT_GROUP(20) +GPIO79 = MODE0(GPIO79) MODE1(I2S0_LRCK) MODE2(PCM1_SYNC) MODE3(I2S3_LRCK) MODE4(I2S1_LRCK) MODE5(PWM3) MODE6(I2S2_LRCK) MODE7(DBG_MON_A27) PD SMT_GROUP(20) +GPIO80 = MODE0(GPIO80) MODE1(I2S0_BCK) MODE2(PCM1_CLK[1]) MODE3(I2S3_BCK) MODE4(I2S1_BCK) MODE5(PWM4) MODE6(I2S2_BCK) MODE7(DBG_MON_A28) PD SMT_GROUP(20) +GPIO81 = MODE0(GPIO81) MODE1(KROW0) MODE2() MODE3(CONN_MCU_DBGI_N) MODE4(CORESONIC_SWCK) MODE5(C2K_TCK) MODE6() MODE7(C2K_DM_EINT1) PD SMT_GROUP(21) +GPIO82 = MODE0(GPIO82) MODE1(KROW1) MODE2() MODE3(CONN_MCU_TRST_B) MODE4(CORESONIC_SWD) MODE5(C2K_NTRST) MODE6(USB_DRVVBUS) MODE7(C2K_DM_EINT2) PD SMT_GROUP(21) +GPIO83 = MODE0(GPIO83) MODE1(KROW2) MODE2(USB_DRVVBUS) MODE3() MODE4() MODE5(C2K_TDI) MODE6() MODE7(C2K_DM_EINT3) PD SMT_GROUP(21) +GPIO84 = MODE0(GPIO84) MODE1(KCOL0) MODE2(URTS0) MODE3(CONN_MCU_DBGACK_N) MODE4(SCL2) MODE5(C2K_TDO) MODE6(AUXIF_CLK) MODE7() PU SMT_GROUP(21) +GPIO85 = MODE0(GPIO85) MODE1(KCOL1) MODE2(UCTS0) MODE3(UCTS1) MODE4(SDA2) MODE5(C2K_TMS) MODE6(AUXIF_ST) MODE7(DBG_MON_A31) PD SMT_GROUP(21) +GPIO86 = MODE0(GPIO86) MODE1(KCOL2) MODE2() MODE3(URTS1) MODE4() MODE5(C2K_RTCK) MODE6() MODE7(DBG_MON_A32) PD SMT_GROUP(21) +GPIO87 = MODE0(GPIO87) MODE1(BPI_BUS5) MODE2(LTE_C2K_BPI_BUS5) MODE3() MODE4() MODE5(C2K_BPI_BUS5) MODE6() MODE7(DBG_MON_B0) PD SMT_GROUP(22) +GPIO88 = MODE0(GPIO88) MODE1(BPI_BUS6) MODE2(LTE_C2K_BPI_BUS6) MODE3() MODE4() MODE5(C2K_BPI_BUS6) MODE6() MODE7(DBG_MON_B1) PD SMT_GROUP(22) +GPIO89 = MODE0(GPIO89) MODE1(BPI_BUS7) MODE2(LTE_C2K_BPI_BUS7) MODE3(CLKM0) MODE4() MODE5(C2K_BPI_BUS7) MODE6() MODE7(DBG_MON_B2) PD SMT_GROUP(22) +GPIO90 = MODE0(GPIO90) MODE1(BPI_BUS8) MODE2(LTE_C2K_BPI_BUS8) MODE3(CLKM1) MODE4() MODE5(C2K_BPI_BUS8) MODE6() MODE7(DBG_MON_B3) PD SMT_GROUP(22) +GPIO91 = MODE0(GPIO91) MODE1(BPI_BUS9) MODE2(LTE_C2K_BPI_BUS9) MODE3(CLKM2) MODE4() MODE5(C2K_BPI_BUS9) MODE6() MODE7(DBG_MON_B4) PD SMT_GROUP(22) +GPIO92 = MODE0(GPIO92) MODE1(BPI_BUS10) MODE2(LTE_C2K_BPI_BUS10) MODE3(CLKM3) MODE4() MODE5(C2K_BPI_BUS10) MODE6() MODE7(DBG_MON_B5) PD SMT_GROUP(22) +GPIO93 = MODE0(GPIO93) MODE1(BPI_BUS11) MODE2(LTE_C2K_BPI_BUS11) MODE3() MODE4() MODE5(C2K_BPI_BUS11) MODE6() MODE7(DBG_MON_B6) PD SMT_GROUP(22) +GPIO94 = MODE0(GPIO94) MODE1(BPI_BUS12) MODE2(LTE_C2K_BPI_BUS12) MODE3() MODE4() MODE5(C2K_BPI_BUS12) MODE6() MODE7(DBG_MON_B7) PD SMT_GROUP(22) +GPIO95 = MODE0(GPIO95) MODE1(BPI_BUS13) MODE2(LTE_C2K_BPI_BUS13) MODE3() MODE4() MODE5(C2K_BPI_BUS13) MODE6() MODE7(DBG_MON_B8) PD SMT_GROUP(22) +GPIO96 = MODE0(GPIO96) MODE1(BPI_BUS14) MODE2(LTE_C2K_BPI_BUS14) MODE3() MODE4() MODE5(C2K_BPI_BUS14) MODE6() MODE7(DBG_MON_B9) PD SMT_GROUP(22) +GPIO97 = MODE0(GPIO97) MODE1(BPI_BUS15) MODE2(LTE_C2K_BPI_BUS15) MODE3() MODE4() MODE5(C2K_BPI_BUS15) MODE6() MODE7(DBG_MON_B10) PD SMT_GROUP(22) +GPIO98 = MODE0(GPIO98) MODE1(BPI_BUS16) MODE2(LTE_C2K_BPI_BUS16) MODE3() MODE4() MODE5(C2K_BPI_BUS16) MODE6() MODE7(DBG_MON_B11) PD SMT_GROUP(22) +GPIO99 = MODE0(GPIO99) MODE1(BPI_BUS17) MODE2(LTE_C2K_BPI_BUS17) MODE3() MODE4() MODE5(C2K_BPI_BUS17) MODE6() MODE7(DBG_MON_B12) PD SMT_GROUP(22) +GPIO100 = MODE0(GPIO100) MODE1(BPI_BUS18) MODE2(LTE_C2K_BPI_BUS18) MODE3() MODE4() MODE5(C2K_BPI_BUS18) MODE6() MODE7(DBG_MON_B13) PD SMT_GROUP(22) +GPIO101 = MODE0(GPIO101) MODE1(BPI_BUS19) MODE2(LTE_C2K_BPI_BUS19) MODE3() MODE4() MODE5(C2K_BPI_BUS19) MODE6() MODE7(DBG_MON_B14) PD SMT_GROUP(22) +GPIO102 = MODE0(GPIO102) MODE1(BPI_BUS20) MODE2(LTE_C2K_BPI_BUS20) MODE3() MODE4() MODE5(C2K_BPI_BUS20) MODE6() MODE7(DBG_MON_B15) PD SMT_GROUP(22) +GPIO103 = MODE0(GPIO103) MODE1(C2K_TXBPI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B16) PD SMT_GROUP(22) +GPIO104 = MODE0(GPIO104) MODE1(RFIC1_BSI_EN) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_EN) MODE6() MODE7(DBG_MON_B17) PD SMT_GROUP(23) +GPIO105 = MODE0(GPIO105) MODE1(RFIC1_BSI_CK) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_CLK) MODE6() MODE7(DBG_MON_B18) PD SMT_GROUP(23) +GPIO106 = MODE0(GPIO106) MODE1(RFIC1_BSI_D0) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_DATA) MODE6() MODE7(DBG_MON_B19) PD SMT_GROUP(23) +GPIO107 = MODE0(GPIO107) MODE1(RFIC1_BSI_D1) MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_EN) MODE6() MODE7(DBG_MON_B20) PD SMT_GROUP(23) +GPIO108 = MODE0(GPIO108) MODE1(RFIC1_BSI_D2) MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_CLK) MODE6() MODE7(DBG_MON_B21) PD SMT_GROUP(23) +GPIO109 = MODE0(GPIO109) MODE1() MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_DATA) MODE6() MODE7(DBG_MON_B22) PD SMT_GROUP(23) +GPIO110 = MODE0(GPIO110) MODE1(RFIC0_BSI_EN) MODE2() MODE3() MODE4(SPM_BSI_EN) MODE5() MODE6() MODE7(DBG_MON_B23) PD SMT_GROUP(23) +GPIO111 = MODE0(GPIO111) MODE1(RFIC0_BSI_CK) MODE2() MODE3() MODE4(SPM_BSI_CLK) MODE5() MODE6() MODE7(DBG_MON_B24) PD SMT_GROUP(23) +GPIO112 = MODE0(GPIO112) MODE1(RFIC0_BSI_D2) MODE2() MODE3() MODE4(SPM_BSI_D2) MODE5() MODE6() MODE7(DBG_MON_B25) PD SMT_GROUP(23) +GPIO113 = MODE0(GPIO113) MODE1(RFIC0_BSI_D1) MODE2() MODE3() MODE4(SPM_BSI_D1) MODE5() MODE6() MODE7(DBG_MON_B26) PD SMT_GROUP(23) +GPIO114 = MODE0(GPIO114) MODE1(RFIC0_BSI_D0) MODE2() MODE3() MODE4(SPM_BSI_D0) MODE5() MODE6() MODE7(DBG_MON_B27) PD SMT_GROUP(23) +GPIO115 = MODE0(GPIO115) MODE1(AUXIN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO116 = MODE0(GPIO116) MODE1(AUXIN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO117 = MODE0(GPIO117) MODE1(AUXIN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO118 = MODE0(GPIO118) MODE1(TXBPI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO119 = MODE0(GPIO119) MODE1(BPI_BUS0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B28) PD SMT_GROUP(24) +GPIO120 = MODE0(GPIO120) MODE1(BPI_BUS1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B29) PD SMT_GROUP(24) +GPIO121 = MODE0(GPIO121) MODE1(BPI_BUS2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B30) PD SMT_GROUP(24) +GPIO122 = MODE0(GPIO122) MODE1(BPI_BUS3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B31) PD SMT_GROUP(24) +GPIO123 = MODE0(GPIO123) MODE1(BPI_BUS4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B32) PD SMT_GROUP(24) +GPIO124 = MODE0(GPIO124) MODE1(BPI_BUS21) MODE2() MODE3() MODE4() MODE5(DPI_HSYNC1) MODE6(KCOL2) MODE7(TDD_TXD) PD SMT_GROUP(24) +GPIO125 = MODE0(GPIO125) MODE1(BPI_BUS22) MODE2() MODE3() MODE4() MODE5(DPI_VSYNC1) MODE6(KROW2) MODE7(MD_URXD) PD SMT_GROUP(24) +GPIO126 = MODE0(GPIO126) MODE1(BPI_BUS23) MODE2() MODE3() MODE4() MODE5(DPI_CK1) MODE6(I2S2_MCK) MODE7(MD_UTXD) PD SMT_GROUP(24) +GPIO127 = MODE0(GPIO127) MODE1(BPI_BUS24) MODE2() MODE3(CONN_MCU_DBGI_N) MODE4(EXT_FRAME_SYNC) MODE5(DPI_DE1) MODE6(SRCLKENAI1) MODE7(URXD0) PD SMT_GROUP(24) +GPIO128 = MODE0(GPIO128) MODE1(BPI_BUS25) MODE2() MODE3(GPS_FRAME_SYNC) MODE4() MODE5(I2S2_DI) MODE6(PTA_RXD) MODE7(UTXD0) PD SMT_GROUP(24) +GPIO129 = MODE0(GPIO129) MODE1(BPI_BUS26) MODE2(DISP_PWM) MODE3() MODE4() MODE5(I2S2_LRCK) MODE6(PTA_TXD) MODE7(LTE_URXD) PD SMT_GROUP(24) +GPIO130 = MODE0(GPIO130) MODE1(BPI_BUS27) MODE2() MODE3() MODE4() MODE5(I2S2_BCK) MODE6(IRTX_OUT) MODE7(LTE_UTXD) PD SMT_GROUP(24) +GPIO131 = MODE0(GPIO131) MODE1(LTE_PAVM0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO132 = MODE0(GPIO132) MODE1(LTE_PAVM1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO133 = MODE0(GPIO133) MODE1(MIPI1_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO134 = MODE0(GPIO134) MODE1(MIPI1_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO135 = MODE0(GPIO135) MODE1(MIPI0_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO136 = MODE0(GPIO136) MODE1(MIPI0_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO137 = MODE0(GPIO137) MODE1(RTC32K_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(25) +GPIO138 = MODE0(GPIO138) MODE1(PWRAP_SPIDO) MODE2(PWRAP_SPIDI) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO139 = MODE0(GPIO139) MODE1(PWRAP_SPIDI) MODE2(PWRAP_SPIDO) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO140 = MODE0(GPIO140) MODE1() MODE2() MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(DM_JTINTP) MODE6() MODE7() PD SMT_GROUP(26) +GPIO141 = MODE0(GPIO141) MODE1(PWRAP_SPICK_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO142 = MODE0(GPIO142) MODE1(PWRAP_SPICS_B_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(26) +GPIO143 = MODE0(GPIO143) MODE1(AUD_CLK_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO144 = MODE0(GPIO144) MODE1(AUD_DAT_MISO) MODE2() MODE3(AUD_DAT_MOSI) MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO145 = MODE0(GPIO145) MODE1(AUD_DAT_MOSI) MODE2() MODE3(AUD_DAT_MISO) MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO146 = MODE0(GPIO146) MODE1(LCM_RST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(28) +GPIO147 = MODE0(GPIO147) MODE1(DSI_TE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(28) +GPIO148 = MODE0(GPIO148) MODE1(SRCLKENA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(29) +GPIO149 = MODE0(GPIO149) MODE1(WATCHDOG) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(25) +GPIO150 = MODE0(GPIO150) MODE1(TDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO151 = MODE0(GPIO151) MODE1(TDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO152 = MODE0(GPIO152) MODE1(TDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO153 = MODE0(GPIO153) MODE1(TDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO154 = MODE0(GPIO154) MODE1(TCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO155 = MODE0(GPIO155) MODE1(TCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO156 = MODE0(GPIO156) MODE1(TDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO157 = MODE0(GPIO157) MODE1(TDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO158 = MODE0(GPIO158) MODE1(TDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO159 = MODE0(GPIO159) MODE1(TDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO160 = MODE0(GPIO160) MODE1(MD_SIM2_SCLK) MODE2(MD_SIM1_SCLK) MODE3(UIM0_CLK) MODE4(UIM1_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO161 = MODE0(GPIO161) MODE1(MD_SIM2_SRST) MODE2(MD_SIM1_SRST) MODE3(UIM0_RST) MODE4(UIM1_RST) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO162 = MODE0(GPIO162) MODE1(MD_SIM2_SDAT) MODE2(MD_SIM1_SDAT) MODE3(UIM0_IO) MODE4(UIM1_IO) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO163 = MODE0(GPIO163) MODE1(MD_SIM1_SCLK) MODE2(MD_SIM2_SCLK) MODE3(UIM1_CLK) MODE4(UIM0_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO164 = MODE0(GPIO164) MODE1(MD_SIM1_SRST) MODE2(MD_SIM2_SRST) MODE3(UIM1_RST) MODE4(UIM0_RST) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO165 = MODE0(GPIO165) MODE1(MD_SIM1_SDAT) MODE2(MD_SIM2_SDAT) MODE3(UIM1_IO) MODE4(UIM0_IO) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO166 = MODE0(GPIO166) MODE1(MSDC1_CMD) MODE2(LTE_MD32_JTAG_TMS) MODE3(C2K_TMS) MODE4(TDD_TMS) MODE5(CONN_DSP_JMS) MODE6(JTMS) MODE7(CONN_MCU_AICE_TMSC) PU SMT_GROUP(32) +GPIO167 = MODE0(GPIO167) MODE1(MSDC1_CLK) MODE2(LTE_MD32_JTAG_TCK) MODE3(C2K_TCK) MODE4(TDD_TCK) MODE5(CONN_DSP_JCK) MODE6(JTCK) MODE7(CONN_MCU_AICE_TCKC) PD SMT_GROUP(33) +GPIO168 = MODE0(GPIO168) MODE1(MSDC1_DAT0) MODE2(LTE_MD32_JTAG_TDI) MODE3(C2K_TDI) MODE4(TDD_TDI) MODE5(CONN_DSP_JDI) MODE6(JTDI) MODE7() PU SMT_GROUP(34) +GPIO169 = MODE0(GPIO169) MODE1(MSDC1_DAT1) MODE2(LTE_MD32_JTAG_TDO) MODE3(C2K_TDO) MODE4(TDD_TDO) MODE5(CONN_DSP_JDO) MODE6(JTDO) MODE7() PU SMT_GROUP(34) +GPIO170 = MODE0(GPIO170) MODE1(MSDC1_DAT2) MODE2(LTE_MD32_JTAG_TRST) MODE3(C2K_NTRST) MODE4(TDD_TRSTN) MODE5(CONN_DSP_JINTP) MODE6(DM_JTINTP) MODE7() PU SMT_GROUP(34) +GPIO171 = MODE0(GPIO171) MODE1(MSDC1_DAT3) MODE2() MODE3(C2K_RTCK) MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(34) +GPIO172 = MODE0(GPIO172) MODE1(MSDC0_CMD) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(35) +GPIO173 = MODE0(GPIO173) MODE1(MSDC0_DSL) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(36) +GPIO174 = MODE0(GPIO174) MODE1(MSDC0_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(37) +GPIO175 = MODE0(GPIO175) MODE1(MSDC0_DAT0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO176 = MODE0(GPIO176) MODE1(MSDC0_DAT1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO177 = MODE0(GPIO177) MODE1(MSDC0_DAT2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO178 = MODE0(GPIO178) MODE1(MSDC0_DAT3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO179 = MODE0(GPIO179) MODE1(MSDC0_DAT4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO180 = MODE0(GPIO180) MODE1(MSDC0_DAT5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO181 = MODE0(GPIO181) MODE1(MSDC0_DAT6) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO182 = MODE0(GPIO182) MODE1(MSDC0_DAT7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO183 = MODE0(GPIO183) MODE1(MSDC0_RSTB) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(39) +GPIO184 = MODE0(GPIO184) MODE1(F2W_DATA) MODE2(MRG_CLK) MODE3(C2K_DM_EINT2) MODE4(PCM0_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(40) +GPIO185 = MODE0(GPIO185) MODE1(F2W_CK) MODE2(MRG_DI) MODE3(C2K_DM_EINT3) MODE4(PCM0_DI) MODE5() MODE6() MODE7() PD SMT_GROUP(40) +GPIO186 = MODE0(GPIO186) MODE1(WB_RSTB) MODE2() MODE3() MODE4(URXD3) MODE5(UTXD3) MODE6() MODE7() PD SMT_GROUP(41) +GPIO187 = MODE0(GPIO187) MODE1(WB_SCLK) MODE2(MRG_DO) MODE3() MODE4(PCM0_DO) MODE5() MODE6() MODE7() PD SMT_GROUP(41) +GPIO188 = MODE0(GPIO188) MODE1(WB_SDATA) MODE2(MRG_SYNC) MODE3() MODE4(PCM0_SYNC) MODE5() MODE6() MODE7() PD SMT_GROUP(41) +GPIO189 = MODE0(GPIO189) MODE1(WB_SEN) MODE2() MODE3() MODE4(UTXD3) MODE5(URXD3) MODE6() MODE7() PD SMT_GROUP(41) +GPIO190 = MODE0(GPIO190) MODE1(GPS_RXQN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO191 = MODE0(GPIO191) MODE1(GPS_RXQP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO192 = MODE0(GPIO192) MODE1(GPS_RXIN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO193 = MODE0(GPIO193) MODE1(GPS_RXIP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO194 = MODE0(GPIO194) MODE1(WB_RXQN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO195 = MODE0(GPIO195) MODE1(WB_RXQP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO196 = MODE0(GPIO196) MODE1(WB_RXIN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO197 = MODE0(GPIO197) MODE1(WB_RXIP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO198 = MODE0(GPIO198) MODE1(MSDC2_CMD) MODE2(SDA1) MODE3(C2K_UART0_RXD) MODE4(C2K_TMS) MODE5(ANT_SEL6) MODE6() MODE7(DM_OTMS) PD SMT_GROUP(42) +GPIO199 = MODE0(GPIO199) MODE1(MSDC2_CLK) MODE2(SCL1) MODE3(C2K_UART0_TXD) MODE4(C2K_TCK) MODE5(ANT_SEL7) MODE6(TDD_TXD) MODE7(DM_OTCK) PD SMT_GROUP(43) +GPIO200 = MODE0(GPIO200) MODE1(MSDC2_DAT0) MODE2(ANT_SEL6) MODE3(GPS_FRAME_SYNC) MODE4(C2K_TDI) MODE5(UTXD0) MODE6() MODE7(DM_OTDI) PD SMT_GROUP(44) +GPIO201 = MODE0(GPIO201) MODE1(MSDC2_DAT1) MODE2(ANT_SEL3) MODE3(PWM0) MODE4(C2K_TDO) MODE5(URXD0) MODE6() MODE7(DM_OTDO) PD SMT_GROUP(44) +GPIO202 = MODE0(GPIO202) MODE1(MSDC2_DAT2) MODE2(ANT_SEL4) MODE3(SDA2) MODE4(C2K_NTRST) MODE5(UTXD1) MODE6(KCOL3) MODE7(DM_JTINTP) PD SMT_GROUP(44) +GPIO203 = MODE0(GPIO203) MODE1(MSDC2_DAT3) MODE2(ANT_SEL5) MODE3(SCL2) MODE4(C2K_RTCK) MODE5(URXD1) MODE6(KCOL6) MODE7() PD SMT_GROUP(44) + +[GPO] + +[EINT] +EINT_COUNT = 213 +EINT_DEBOUNCE_TIME_COUNT = 12 + +[WAKE_UP_SRC] +ACCDET +MSDC1_INS +MSDC2_INS +CHR_STAT +IRQ_NFC +COMBO_BGF +WIFI +TOUCH_PANEL +ALS +MHL +EXT_BUCK_OC + +[EINT_MD1] +EINT_MD1_COUNT = 4 + +[ADC] +ADC_COUNT = 6 + +[ADC_EX_PIN] +0 +1 +12 +13 +14 +15 + +[KEYPAD] +KEY_ROW = 8 +KEY_COLUMN = 9 + +[KEYPAD_EXTEND_TYPE] +KEY_ROW = 3 +KEY_COLUMN = 6 +KEY_DESIGN_NOTICES = "Please notice that the circuit design are different between single and double keypad, and DO NOT add 1K ohm resistor on double keypad" + +[I2C] +I2C_COUNT = 32 +CHANNEL_COUNT = 4 + +[CLK_BUF] +CLK_BUF_COUNT = 4 + +[POWER] +DVDD28_SIM1 +DVDD28_SIM2 +DVDD28_MC1 +DVDD28_MC2 + +[DVDD28_SIM1] +VIO18 +VIO28 + +[DVDD28_SIM2] +VIO18 +VIO28 + +[DVDD28_MC1] +VIO18 +VIO28 + +[DVDD28_MC2] +VIO18 +VIO28 + +[MSDC_POWER_MC1] +MSDC_VIO18_MC1 +MSDC_VIO28_MC1 +MSDC_VMC + + +[SRC_PIN] +PAD_ANT_SEL1 +PAD_ANT_SEL2 +PAD_CMDAT1 +PAD_EINT1 +PAD_EINT2 +PAD_EINT3 +PAD_EINT8 +PAD_EINT9 +PAD_EINT10 +PAD_EINT11 +PAD_EINT12 +PAD_SCL3 +PAD_SDA3 + +[SRC_PIN_INDEX] +2 +1 +0 +0 +0 +0 +1 +2 +0 +1 +2 +1 +2 + diff --git a/tools/dct/MT6735M.fig b/tools/dct/MT6735M.fig new file mode 100755 index 000000000..bdcf66e27 --- /dev/null +++ b/tools/dct/MT6735M.fig @@ -0,0 +1,327 @@ +[Chip Type] +Chip = MT6735 +GPIO_Pull_Sel = 1 +PMIC_Config = 1 +PMIC_ON_OFF_CONFIG = 1 +GPIO_ModeNum = 8 +EINT_MD1_Config = 1 +EINT_MD1_SRC_PIN = 1 +POWER_Config = 1 +POWER_COUNT = 4 +SpecialKey_Config = 1 +I2C_Config = 1 +CLOCK_BUFFER_CONFIG = 1 +Extend_Key_Config=1 +Reset_Key_Config = 1 +PMIC_APP_Ver = 2 +PMIC_APP_COUNT = 6 +MD1_EINT_SRC_PIN_CFG = 1 +AndroidPhone = 1 + +[GPIO] +GPIO0 = MODE0(GPIO0) MODE1(IDDIG) MODE2(DPI_D4) MODE3(CLKM4) MODE4(EXT_FRAME_SYNC) MODE5(PWM3) MODE6(KCOL2) MODE7(C2K_ARM_EINT0) PD SMT_GROUP(1) +GPIO1 = MODE0(GPIO1) MODE1(PWM2) MODE2(DPI_D5) MODE3(MD_EINT0) MODE4(TDD_TDO) MODE5(CONN_MCU_TDO) MODE6(PTA_RXD) MODE7(C2K_ARM_EINT1) PD SMT_GROUP(1) +GPIO2 = MODE0(GPIO2) MODE1(CLKM0) MODE2(DPI_D6) MODE3(MD_EINT0) MODE4(USB_DRVVBUS) MODE5(CONN_MCU_DBGACK_N) MODE6(PTA_TXD) MODE7(C2K_ARM_EINT2) PD SMT_GROUP(1) +GPIO3 = MODE0(GPIO3) MODE1(CLKM1) MODE2(DPI_D7) MODE3(SPI_MIB) MODE4(MD_EINT0) MODE5(CONN_MCU_DBGI_N) MODE6(CONN_MCU_AICE_TMSC) MODE7(C2K_ARM_EINT3) PD SMT_GROUP(1) +GPIO4 = MODE0(GPIO4) MODE1(CLKM2) MODE2(DPI_D8) MODE3(SPI_MOB) MODE4(TDD_TCK) MODE5(CONN_MCU_TCK[0]) MODE6(CONN_MCU_AICE_TCKC) MODE7(C2K_DM_EINT0) PD SMT_GROUP(1) +GPIO5 = MODE0(GPIO5) MODE1(UCTS2) MODE2(DPI_D9) MODE3(SPI_CSB) MODE4(TDD_TDI) MODE5(CONN_MCU_TDI) MODE6(I2S1_DO) MODE7(MD_URXD) PD SMT_GROUP(2) +GPIO6 = MODE0(GPIO6) MODE1(URTS2) MODE2(DPI_D10) MODE3(SPI_CKB) MODE4(TDD_TRSTN) MODE5(CONN_MCU_TRST_B) MODE6(I2S1_LRCK) MODE7(MD_UTXD) PD SMT_GROUP(2) +GPIO7 = MODE0(GPIO7) MODE1(UCTS3) MODE2(DPI_D11) MODE3(SDA1) MODE4(TDD_TMS) MODE5(CONN_MCU_TMS) MODE6(I2S1_BCK) MODE7(TDD_TXD) PD SMT_GROUP(2) +GPIO8 = MODE0(GPIO8) MODE1(URTS3) MODE2(C2K_UIM0_HOT_PLUG_IN) MODE3(SCL1) MODE4(PCM1_DO1) MODE5(MD_EINT1) MODE6(KCOL4) MODE7(UTXD0) PD SMT_GROUP(2) +GPIO9 = MODE0(GPIO9) MODE1(C2K_UIM1_HOT_PLUG_IN) MODE2(PCM1_DO0) MODE3(I2S3_MCK) MODE4(MD_EINT2) MODE5(CLKM2) MODE6(I2S1_MCK) MODE7(DBG_MON_A29) PD SMT_GROUP(3) +GPIO10 = MODE0(GPIO10) MODE1(PWM1) MODE2(CLKM1) MODE3(KROW2) MODE4(MD_EINT0) MODE5(I2S1_MCK) MODE6(SDA3) MODE7(DBG_MON_A30) PD SMT_GROUP(3) +GPIO11 = MODE0(GPIO11) MODE1(MD_EINT1) MODE2(IRTX_OUT) MODE3(C2K_UIM0_HOT_PLUG_IN) MODE4(CLKM0) MODE5(I2S2_MCK) MODE6(SCL3) MODE7(URXD0) PD SMT_GROUP(3) +GPIO12 = MODE0(GPIO12) MODE1(I2S0_MCK) MODE2(C2K_UIM1_HOT_PLUG_IN) MODE3(KCOL2) MODE4(MD_EINT2) MODE5(IRTX_OUT) MODE6(SRCLKENAI2) MODE7(PCM1_DO1) PD SMT_GROUP(3) +GPIO13 = MODE0(GPIO13) MODE1(WB_CTRL0) MODE2() MODE3(C2K_ARM_EINT0) MODE4() MODE5() MODE6() MODE7(DBG_MON_A0) PD SMT_GROUP(4) +GPIO14 = MODE0(GPIO14) MODE1(WB_CTRL1) MODE2() MODE3(C2K_ARM_EINT1) MODE4() MODE5() MODE6() MODE7(DBG_MON_A1) PD SMT_GROUP(4) +GPIO15 = MODE0(GPIO15) MODE1(WB_CTRL2) MODE2() MODE3(C2K_ARM_EINT2) MODE4() MODE5() MODE6() MODE7(DBG_MON_A2) PD SMT_GROUP(4) +GPIO16 = MODE0(GPIO16) MODE1(WB_CTRL3) MODE2() MODE3(C2K_ARM_EINT3) MODE4() MODE5() MODE6() MODE7(DBG_MON_A3) PD SMT_GROUP(4) +GPIO17 = MODE0(GPIO17) MODE1(WB_CTRL4) MODE2() MODE3(C2K_DM_EINT0) MODE4(WATCHDOG) MODE5() MODE6() MODE7(DBG_MON_A4) PD SMT_GROUP(4) +GPIO18 = MODE0(GPIO18) MODE1(WB_CTRL5) MODE2() MODE3(C2K_DM_EINT1) MODE4() MODE5() MODE6() MODE7(DBG_MON_A5) PD SMT_GROUP(4) +GPIO19 = MODE0(GPIO19) MODE1(ANT_SEL0) MODE2(IRTX_OUT) MODE3(IRDA_TX) MODE4(C2K_UART0_TXD) MODE5(GPS_FRAME_SYNC) MODE6(LTE_UTXD) MODE7(DBG_MON_A6) PD SMT_GROUP(5) +GPIO20 = MODE0(GPIO20) MODE1(ANT_SEL1) MODE2(C2K_UIM1_HOT_PLUG_IN) MODE3(IRDA_RX) MODE4(C2K_UART0_RXD) MODE5(MD_EINT2) MODE6(LTE_URXD) MODE7(DBG_MON_A7) PD SMT_GROUP(5) +GPIO21 = MODE0(GPIO21) MODE1(ANT_SEL2) MODE2(PWM2) MODE3(IRDA_PDN) MODE4(CORESONIC_SWCK) MODE5(MD_EINT1) MODE6(C2K_UIM0_HOT_PLUG_IN) MODE7(DBG_MON_A8) PD SMT_GROUP(5) +GPIO22 = MODE0(GPIO22) MODE1(RDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO23 = MODE0(GPIO23) MODE1(RDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO24 = MODE0(GPIO24) MODE1(RDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO25 = MODE0(GPIO25) MODE1(RDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO26 = MODE0(GPIO26) MODE1(RCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO27 = MODE0(GPIO27) MODE1(RCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO28 = MODE0(GPIO28) MODE1(RDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO29 = MODE0(GPIO29) MODE1(RDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO30 = MODE0(GPIO30) MODE1(RDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO31 = MODE0(GPIO31) MODE1(RDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO32 = MODE0(GPIO32) MODE1(RDN0_A) MODE2(CMHSYNC) MODE3(CMCSD0) MODE4() MODE5() MODE6() MODE7() -- +GPIO33 = MODE0(GPIO33) MODE1(RDP0_A) MODE2(CMVSYNC) MODE3(CMCSD1) MODE4() MODE5() MODE6() MODE7() -- +GPIO34 = MODE0(GPIO34) MODE1(RDN1_A) MODE2(CMDAT9) MODE3(CMCSD2) MODE4() MODE5() MODE6() MODE7() -- +GPIO35 = MODE0(GPIO35) MODE1(RDP1_A) MODE2(CMDAT8) MODE3(CMCSD3) MODE4() MODE5() MODE6() MODE7() -- +GPIO36 = MODE0(GPIO36) MODE1(RCN_A) MODE2(CMDAT7) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO37 = MODE0(GPIO37) MODE1(RCP_A) MODE2(CMDAT6) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO38 = MODE0(GPIO38) MODE1(RDN2_A) MODE2(CMDAT5) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO39 = MODE0(GPIO39) MODE1(RDP2_A) MODE2(CMDAT4) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO40 = MODE0(GPIO40) MODE1(RDN3_A) MODE2(CMDAT3) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO41 = MODE0(GPIO41) MODE1(RDP3_A) MODE2(CMDAT2) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO42 = MODE0(GPIO42) MODE1(CMDAT0) MODE2(CMCSD0) MODE3(CMMCLK1) MODE4() MODE5(ANT_SEL5) MODE6(CLKM5) MODE7(DBG_MON_A9) PD SMT_GROUP(6) +GPIO43 = MODE0(GPIO43) MODE1(CMDAT1) MODE2(CMCSD1) MODE3(CMFLASH) MODE4(MD_EINT0) MODE5(CMMCLK1) MODE6(CLKM4) MODE7(DBG_MON_A10) PD SMT_GROUP(6) +GPIO44 = MODE0(GPIO44) MODE1(CMPCLK) MODE2(CMCSK) MODE3(CMCSD2) MODE4(KCOL3) MODE5(SRCLKENAI2) MODE6(PWM0) MODE7(DBG_MON_A11) PD SMT_GROUP(6) +GPIO45 = MODE0(GPIO45) MODE1(CMMCLK0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A12) PD SMT_GROUP(7) +GPIO46 = MODE0(GPIO46) MODE1(CMMCLK1) MODE2(IDDIG) MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(DM_JTINTP) MODE6(KCOL6) MODE7(DBG_MON_A13) PD SMT_GROUP(7) +GPIO47 = MODE0(GPIO47) MODE1(SDA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(8) +GPIO48 = MODE0(GPIO48) MODE1(SCL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(8) +GPIO49 = MODE0(GPIO49) MODE1(SDA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(9) +GPIO50 = MODE0(GPIO50) MODE1(SCL1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(9) +GPIO51 = MODE0(GPIO51) MODE1(SDA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(10) +GPIO52 = MODE0(GPIO52) MODE1(SCL2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(10) +GPIO53 = MODE0(GPIO53) MODE1(SDA3) MODE2() MODE3(IDDIG) MODE4() MODE5(MD_EINT2) MODE6(C2K_UIM1_HOT_PLUG_IN) MODE7() PU SMT_GROUP(11) +GPIO54 = MODE0(GPIO54) MODE1(SCL3) MODE2() MODE3(IDDIG) MODE4() MODE5(MD_EINT1) MODE6(C2K_UIM0_HOT_PLUG_IN) MODE7() PU SMT_GROUP(11) +GPIO55 = MODE0(GPIO55) MODE1(SRCLKENAI0) MODE2(PWM2) MODE3(CLKM5) MODE4(CORESONIC_SWD) MODE5(ANT_SEL6) MODE6(KROW5) MODE7(DISP_PWM) PD SMT_GROUP(12) +GPIO56 = MODE0(GPIO56) MODE1(SRCLKENA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(13) +GPIO57 = MODE0(GPIO57) MODE1(URXD2) MODE2(DPI_HSYNC0) MODE3(UTXD2) MODE4(MD_URXD) MODE5(SRCLKENAI1) MODE6(KROW4) MODE7(DBG_MON_A14) PD SMT_GROUP(14) +GPIO58 = MODE0(GPIO58) MODE1(UTXD2) MODE2(DPI_VSYNC0) MODE3(URXD2) MODE4(MD_UTXD) MODE5(TDD_TXD) MODE6(KROW5) MODE7(DBG_MON_A15) PD SMT_GROUP(14) +GPIO59 = MODE0(GPIO59) MODE1(URXD3) MODE2(DPI_CK0) MODE3(UTXD3) MODE4(UCTS2) MODE5(PWM3) MODE6(KROW6) MODE7(DBG_MON_A16) PD SMT_GROUP(14) +GPIO60 = MODE0(GPIO60) MODE1(UTXD3) MODE2(DPI_DE0) MODE3(URXD3) MODE4(URTS2) MODE5(PWM4) MODE6(KROW7) MODE7(DBG_MON_A17) PD SMT_GROUP(14) +GPIO61 = MODE0(GPIO61) MODE1(PCM1_CLK[0]) MODE2(DPI_D0) MODE3(I2S0_BCK) MODE4(KROW4) MODE5(ANT_SEL3) MODE6(IRTX_OUT) MODE7(DBG_MON_A18) PD SMT_GROUP(15) +GPIO62 = MODE0(GPIO62) MODE1(PCM1_SYNC) MODE2(DPI_D1) MODE3(I2S0_LRCK) MODE4(KCOL7) MODE5(CLKM3) MODE6(CMFLASH) MODE7(DBG_MON_A19) PD SMT_GROUP(15) +GPIO63 = MODE0(GPIO63) MODE1(PCM1_DI) MODE2(DPI_D2) MODE3(I2S0_DI) MODE4(PCM1_DO0) MODE5(CLKM5) MODE6(KROW3) MODE7(DBG_MON_A20) PD SMT_GROUP(15) +GPIO64 = MODE0(GPIO64) MODE1(PCM1_DO0) MODE2(DPI_D3) MODE3(I2S0_MCK) MODE4(PCM1_DI) MODE5(SRCLKENAI2) MODE6(KCOL5) MODE7(DBG_MON_A21) PD SMT_GROUP(15) +GPIO65 = MODE0(GPIO65) MODE1(SPI_CSA) MODE2(EXT_FRAME_SYNC) MODE3(I2S3_MCK) MODE4(KROW2) MODE5(GPS_FRAME_SYNC) MODE6(PTA_RXD) MODE7(DBG_MON_A22) PD SMT_GROUP(16) +GPIO66 = MODE0(GPIO66) MODE1(SPI_CKA) MODE2(USB_DRVVBUS) MODE3(I2S3_BCK) MODE4(KCOL2) MODE5() MODE6(PTA_TXD) MODE7(DBG_MON_A23) PD SMT_GROUP(16) +GPIO67 = MODE0(GPIO67) MODE1(SPI_MIA) MODE2(SPI_MOA) MODE3(I2S3_DO) MODE4(PTA_RXD) MODE5(IDDIG) MODE6(UCTS1) MODE7(DBG_MON_A24) PD SMT_GROUP(16) +GPIO68 = MODE0(GPIO68) MODE1(SPI_MOA) MODE2(SPI_MIA) MODE3(I2S3_LRCK) MODE4(PTA_TXD) MODE5(ANT_SEL4) MODE6(URTS1) MODE7(DBG_MON_A25) PD SMT_GROUP(16) +GPIO69 = MODE0(GPIO69) MODE1(DISP_PWM) MODE2(PWM1) MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(ANT_SEL7) MODE6(DM_JTINTP) MODE7() PD SMT_GROUP(17) +GPIO70 = MODE0(GPIO70) MODE1(JTMS) MODE2(CONN_MCU_TMS) MODE3(LTE_MD32_JTAG_TMS) MODE4(TDD_TMS) MODE5(CORESONIC_SWD) MODE6(DM_OTMS) MODE7(DFD_TMS) PU SMT_GROUP(18) +GPIO71 = MODE0(GPIO71) MODE1(JTCK) MODE2(CONN_MCU_TCK[1]) MODE3(LTE_MD32_JTAG_TCK) MODE4(TDD_TCK) MODE5(CORESONIC_SWCK) MODE6(DM_OTCK) MODE7(DFD_TCK_XI) PU SMT_GROUP(18) +GPIO72 = MODE0(GPIO72) MODE1(JTDI) MODE2(CONN_MCU_TDI) MODE3(LTE_MD32_JTAG_TDI) MODE4(TDD_TDI) MODE5() MODE6(DM_OTDI) MODE7(DFD_TDI) PU SMT_GROUP(18) +GPIO73 = MODE0(GPIO73) MODE1(JTDO) MODE2(CONN_MCU_TDO) MODE3(LTE_MD32_JTAG_TDO) MODE4(TDD_TDO) MODE5() MODE6(DM_OTDO) MODE7(DFD_TDO) PU SMT_GROUP(18) +GPIO74 = MODE0(GPIO74) MODE1(URXD0) MODE2(UTXD0) MODE3(MD_URXD) MODE4(SDA3) MODE5(C2K_UART0_RXD) MODE6(LTE_URXD) MODE7(AUXIF_ST) PU SMT_GROUP(19) +GPIO75 = MODE0(GPIO75) MODE1(UTXD0) MODE2(URXD0) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5(C2K_UART0_TXD) MODE6(LTE_UTXD) MODE7() PU SMT_GROUP(19) +GPIO76 = MODE0(GPIO76) MODE1(URXD1) MODE2(UTXD1) MODE3(MD_URXD) MODE4(SCL3) MODE5(LTE_URXD) MODE6(C2K_UART0_RXD) MODE7(AUXIF_CLK) PD SMT_GROUP(19) +GPIO77 = MODE0(GPIO77) MODE1(UTXD1) MODE2(URXD1) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5(LTE_UTXD) MODE6(C2K_UART0_TXD) MODE7() PD SMT_GROUP(19) +GPIO78 = MODE0(GPIO78) MODE1(I2S0_DI) MODE2(PCM1_DI) MODE3(I2S3_DO) MODE4(I2S1_DO) MODE5(PWM0) MODE6(I2S2_DI) MODE7(DBG_MON_A26) PD SMT_GROUP(20) +GPIO79 = MODE0(GPIO79) MODE1(I2S0_LRCK) MODE2(PCM1_SYNC) MODE3(I2S3_LRCK) MODE4(I2S1_LRCK) MODE5(PWM3) MODE6(I2S2_LRCK) MODE7(DBG_MON_A27) PD SMT_GROUP(20) +GPIO80 = MODE0(GPIO80) MODE1(I2S0_BCK) MODE2(PCM1_CLK[1]) MODE3(I2S3_BCK) MODE4(I2S1_BCK) MODE5(PWM4) MODE6(I2S2_BCK) MODE7(DBG_MON_A28) PD SMT_GROUP(20) +GPIO81 = MODE0(GPIO81) MODE1(KROW0) MODE2() MODE3(CONN_MCU_DBGI_N) MODE4(CORESONIC_SWCK) MODE5(C2K_TCK) MODE6() MODE7(C2K_DM_EINT1) PD SMT_GROUP(21) +GPIO82 = MODE0(GPIO82) MODE1(KROW1) MODE2() MODE3(CONN_MCU_TRST_B) MODE4(CORESONIC_SWD) MODE5(C2K_NTRST) MODE6(USB_DRVVBUS) MODE7(C2K_DM_EINT2) PD SMT_GROUP(21) +GPIO83 = MODE0(GPIO83) MODE1(KROW2) MODE2(USB_DRVVBUS) MODE3() MODE4() MODE5(C2K_TDI) MODE6() MODE7(C2K_DM_EINT3) PD SMT_GROUP(21) +GPIO84 = MODE0(GPIO84) MODE1(KCOL0) MODE2(URTS0) MODE3(CONN_MCU_DBGACK_N) MODE4(SCL2) MODE5(C2K_TDO) MODE6(AUXIF_CLK) MODE7() PU SMT_GROUP(21) +GPIO85 = MODE0(GPIO85) MODE1(KCOL1) MODE2(UCTS0) MODE3(UCTS1) MODE4(SDA2) MODE5(C2K_TMS) MODE6(AUXIF_ST) MODE7(DBG_MON_A31) PD SMT_GROUP(21) +GPIO86 = MODE0(GPIO86) MODE1(KCOL2) MODE2() MODE3(URTS1) MODE4() MODE5(C2K_RTCK) MODE6() MODE7(DBG_MON_A32) PD SMT_GROUP(21) +GPIO87 = MODE0(GPIO87) MODE1(BPI_BUS5) MODE2(LTE_C2K_BPI_BUS5) MODE3() MODE4() MODE5(C2K_BPI_BUS5) MODE6() MODE7(DBG_MON_B0) PD SMT_GROUP(22) +GPIO88 = MODE0(GPIO88) MODE1(BPI_BUS6) MODE2(LTE_C2K_BPI_BUS6) MODE3() MODE4() MODE5(C2K_BPI_BUS6) MODE6() MODE7(DBG_MON_B1) PD SMT_GROUP(22) +GPIO89 = MODE0(GPIO89) MODE1(BPI_BUS7) MODE2(LTE_C2K_BPI_BUS7) MODE3(CLKM0) MODE4() MODE5(C2K_BPI_BUS7) MODE6() MODE7(DBG_MON_B2) PD SMT_GROUP(22) +GPIO90 = MODE0(GPIO90) MODE1(BPI_BUS8) MODE2(LTE_C2K_BPI_BUS8) MODE3(CLKM1) MODE4() MODE5(C2K_BPI_BUS8) MODE6() MODE7(DBG_MON_B3) PD SMT_GROUP(22) +GPIO91 = MODE0(GPIO91) MODE1(BPI_BUS9) MODE2(LTE_C2K_BPI_BUS9) MODE3(CLKM2) MODE4() MODE5(C2K_BPI_BUS9) MODE6() MODE7(DBG_MON_B4) PD SMT_GROUP(22) +GPIO92 = MODE0(GPIO92) MODE1(BPI_BUS10) MODE2(LTE_C2K_BPI_BUS10) MODE3(CLKM3) MODE4() MODE5(C2K_BPI_BUS10) MODE6() MODE7(DBG_MON_B5) PD SMT_GROUP(22) +GPIO93 = MODE0(GPIO93) MODE1(BPI_BUS11) MODE2(LTE_C2K_BPI_BUS11) MODE3() MODE4() MODE5(C2K_BPI_BUS11) MODE6() MODE7(DBG_MON_B6) PD SMT_GROUP(22) +GPIO94 = MODE0(GPIO94) MODE1(BPI_BUS12) MODE2(LTE_C2K_BPI_BUS12) MODE3() MODE4() MODE5(C2K_BPI_BUS12) MODE6() MODE7(DBG_MON_B7) PD SMT_GROUP(22) +GPIO95 = MODE0(GPIO95) MODE1(BPI_BUS13) MODE2(LTE_C2K_BPI_BUS13) MODE3() MODE4() MODE5(C2K_BPI_BUS13) MODE6() MODE7(DBG_MON_B8) PD SMT_GROUP(22) +GPIO96 = MODE0(GPIO96) MODE1(BPI_BUS14) MODE2(LTE_C2K_BPI_BUS14) MODE3() MODE4() MODE5(C2K_BPI_BUS14) MODE6() MODE7(DBG_MON_B9) PD SMT_GROUP(22) +GPIO97 = MODE0(GPIO97) MODE1(BPI_BUS15) MODE2(LTE_C2K_BPI_BUS15) MODE3() MODE4() MODE5(C2K_BPI_BUS15) MODE6() MODE7(DBG_MON_B10) PD SMT_GROUP(22) +GPIO98 = MODE0(GPIO98) MODE1(BPI_BUS16) MODE2(LTE_C2K_BPI_BUS16) MODE3() MODE4() MODE5(C2K_BPI_BUS16) MODE6() MODE7(DBG_MON_B11) PD SMT_GROUP(22) +GPIO99 = MODE0(GPIO99) MODE1(BPI_BUS17) MODE2(LTE_C2K_BPI_BUS17) MODE3() MODE4() MODE5(C2K_BPI_BUS17) MODE6() MODE7(DBG_MON_B12) PD SMT_GROUP(22) +GPIO100 = MODE0(GPIO100) MODE1(BPI_BUS18) MODE2(LTE_C2K_BPI_BUS18) MODE3() MODE4() MODE5(C2K_BPI_BUS18) MODE6() MODE7(DBG_MON_B13) PD SMT_GROUP(22) +GPIO101 = MODE0(GPIO101) MODE1(BPI_BUS19) MODE2(LTE_C2K_BPI_BUS19) MODE3() MODE4() MODE5(C2K_BPI_BUS19) MODE6() MODE7(DBG_MON_B14) PD SMT_GROUP(22) +GPIO102 = MODE0(GPIO102) MODE1(BPI_BUS20) MODE2(LTE_C2K_BPI_BUS20) MODE3() MODE4() MODE5(C2K_BPI_BUS20) MODE6() MODE7(DBG_MON_B15) PD SMT_GROUP(22) +GPIO103 = MODE0(GPIO103) MODE1(C2K_TXBPI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B16) PD SMT_GROUP(22) +GPIO104 = MODE0(GPIO104) MODE1(RFIC1_BSI_EN) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_EN) MODE6() MODE7(DBG_MON_B17) PD SMT_GROUP(23) +GPIO105 = MODE0(GPIO105) MODE1(RFIC1_BSI_CK) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_CLK) MODE6() MODE7(DBG_MON_B18) PD SMT_GROUP(23) +GPIO106 = MODE0(GPIO106) MODE1(RFIC1_BSI_D0) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_DATA) MODE6() MODE7(DBG_MON_B19) PD SMT_GROUP(23) +GPIO107 = MODE0(GPIO107) MODE1(RFIC1_BSI_D1) MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_EN) MODE6() MODE7(DBG_MON_B20) PD SMT_GROUP(23) +GPIO108 = MODE0(GPIO108) MODE1(RFIC1_BSI_D2) MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_CLK) MODE6() MODE7(DBG_MON_B21) PD SMT_GROUP(23) +GPIO109 = MODE0(GPIO109) MODE1() MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_DATA) MODE6() MODE7(DBG_MON_B22) PD SMT_GROUP(23) +GPIO110 = MODE0(GPIO110) MODE1(RFIC0_BSI_EN) MODE2() MODE3() MODE4(SPM_BSI_EN) MODE5() MODE6() MODE7(DBG_MON_B23) PD SMT_GROUP(23) +GPIO111 = MODE0(GPIO111) MODE1(RFIC0_BSI_CK) MODE2() MODE3() MODE4(SPM_BSI_CLK) MODE5() MODE6() MODE7(DBG_MON_B24) PD SMT_GROUP(23) +GPIO112 = MODE0(GPIO112) MODE1(RFIC0_BSI_D2) MODE2() MODE3() MODE4(SPM_BSI_D2) MODE5() MODE6() MODE7(DBG_MON_B25) PD SMT_GROUP(23) +GPIO113 = MODE0(GPIO113) MODE1(RFIC0_BSI_D1) MODE2() MODE3() MODE4(SPM_BSI_D1) MODE5() MODE6() MODE7(DBG_MON_B26) PD SMT_GROUP(23) +GPIO114 = MODE0(GPIO114) MODE1(RFIC0_BSI_D0) MODE2() MODE3() MODE4(SPM_BSI_D0) MODE5() MODE6() MODE7(DBG_MON_B27) PD SMT_GROUP(23) +GPIO115 = MODE0(GPIO115) MODE1(AUXIN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO116 = MODE0(GPIO116) MODE1(AUXIN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO117 = MODE0(GPIO117) MODE1(AUXIN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO118 = MODE0(GPIO118) MODE1(TXBPI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO119 = MODE0(GPIO119) MODE1(BPI_BUS0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B28) PD SMT_GROUP(24) +GPIO120 = MODE0(GPIO120) MODE1(BPI_BUS1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B29) PD SMT_GROUP(24) +GPIO121 = MODE0(GPIO121) MODE1(BPI_BUS2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B30) PD SMT_GROUP(24) +GPIO122 = MODE0(GPIO122) MODE1(BPI_BUS3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B31) PD SMT_GROUP(24) +GPIO123 = MODE0(GPIO123) MODE1(BPI_BUS4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B32) PD SMT_GROUP(24) +GPIO124 = MODE0(GPIO124) MODE1(BPI_BUS21) MODE2() MODE3() MODE4() MODE5(DPI_HSYNC1) MODE6(KCOL2) MODE7(TDD_TXD) PD SMT_GROUP(24) +GPIO125 = MODE0(GPIO125) MODE1(BPI_BUS22) MODE2() MODE3() MODE4() MODE5(DPI_VSYNC1) MODE6(KROW2) MODE7(MD_URXD) PD SMT_GROUP(24) +GPIO126 = MODE0(GPIO126) MODE1(BPI_BUS23) MODE2() MODE3() MODE4() MODE5(DPI_CK1) MODE6(I2S2_MCK) MODE7(MD_UTXD) PD SMT_GROUP(24) +GPIO127 = MODE0(GPIO127) MODE1(BPI_BUS24) MODE2() MODE3(CONN_MCU_DBGI_N) MODE4(EXT_FRAME_SYNC) MODE5(DPI_DE1) MODE6(SRCLKENAI1) MODE7(URXD0) PD SMT_GROUP(24) +GPIO128 = MODE0(GPIO128) MODE1(BPI_BUS25) MODE2(SDA3) MODE3(GPS_FRAME_SYNC) MODE4() MODE5(I2S2_DI) MODE6(PTA_RXD) MODE7(UTXD0) PD SMT_GROUP(24) +GPIO129 = MODE0(GPIO129) MODE1(BPI_BUS26) MODE2(DISP_PWM) MODE3(SCL3) MODE4() MODE5(I2S2_LRCK) MODE6(PTA_TXD) MODE7(LTE_URXD) PD SMT_GROUP(24) +GPIO130 = MODE0(GPIO130) MODE1(BPI_BUS27) MODE2() MODE3() MODE4() MODE5(I2S2_BCK) MODE6(IRTX_OUT) MODE7(LTE_UTXD) PD SMT_GROUP(24) +GPIO131 = MODE0(GPIO131) MODE1(LTE_PAVM0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO132 = MODE0(GPIO132) MODE1(LTE_PAVM1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO133 = MODE0(GPIO133) MODE1(MIPI1_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO134 = MODE0(GPIO134) MODE1(MIPI1_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO135 = MODE0(GPIO135) MODE1(MIPI0_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO136 = MODE0(GPIO136) MODE1(MIPI0_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO137 = MODE0(GPIO137) MODE1(RTC32K_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(25) +GPIO138 = MODE0(GPIO138) MODE1(PWRAP_SPIDO) MODE2(PWRAP_SPIDI) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO139 = MODE0(GPIO139) MODE1(PWRAP_SPIDI) MODE2(PWRAP_SPIDO) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO140 = MODE0(GPIO140) MODE1() MODE2() MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(DM_JTINTP) MODE6() MODE7() PD SMT_GROUP(26) +GPIO141 = MODE0(GPIO141) MODE1(PWRAP_SPICK_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO142 = MODE0(GPIO142) MODE1(PWRAP_SPICS_B_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(26) +GPIO143 = MODE0(GPIO143) MODE1(AUD_CLK_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO144 = MODE0(GPIO144) MODE1(AUD_DAT_MISO) MODE2() MODE3(AUD_DAT_MOSI) MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO145 = MODE0(GPIO145) MODE1(AUD_DAT_MOSI) MODE2() MODE3(AUD_DAT_MISO) MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO146 = MODE0(GPIO146) MODE1(LCM_RST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(28) +GPIO147 = MODE0(GPIO147) MODE1(DSI_TE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(28) +GPIO148 = MODE0(GPIO148) MODE1(SRCLKENA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(29) +GPIO149 = MODE0(GPIO149) MODE1(WATCHDOG) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(25) +GPIO150 = MODE0(GPIO150) MODE1(TDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO151 = MODE0(GPIO151) MODE1(TDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO152 = MODE0(GPIO152) MODE1(TDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO153 = MODE0(GPIO153) MODE1(TDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO154 = MODE0(GPIO154) MODE1(TCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO155 = MODE0(GPIO155) MODE1(TCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO156 = MODE0(GPIO156) MODE1(TDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO157 = MODE0(GPIO157) MODE1(TDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO158 = MODE0(GPIO158) MODE1(TDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO159 = MODE0(GPIO159) MODE1(TDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO160 = MODE0(GPIO160) MODE1(MD_SIM2_SCLK) MODE2(MD_SIM1_SCLK) MODE3(UIM0_CLK) MODE4(UIM1_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO161 = MODE0(GPIO161) MODE1(MD_SIM2_SRST) MODE2(MD_SIM1_SRST) MODE3(UIM0_RST) MODE4(UIM1_RST) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO162 = MODE0(GPIO162) MODE1(MD_SIM2_SDAT) MODE2(MD_SIM1_SDAT) MODE3(UIM0_IO) MODE4(UIM1_IO) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO163 = MODE0(GPIO163) MODE1(MD_SIM1_SCLK) MODE2(MD_SIM2_SCLK) MODE3(UIM1_CLK) MODE4(UIM0_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO164 = MODE0(GPIO164) MODE1(MD_SIM1_SRST) MODE2(MD_SIM2_SRST) MODE3(UIM1_RST) MODE4(UIM0_RST) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO165 = MODE0(GPIO165) MODE1(MD_SIM1_SDAT) MODE2(MD_SIM2_SDAT) MODE3(UIM1_IO) MODE4(UIM0_IO) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO166 = MODE0(GPIO166) MODE1(MSDC1_CMD) MODE2(LTE_MD32_JTAG_TMS) MODE3(C2K_TMS) MODE4(TDD_TMS) MODE5(CONN_DSP_JMS) MODE6(JTMS) MODE7(CONN_MCU_AICE_TMSC) PU SMT_GROUP(32) +GPIO167 = MODE0(GPIO167) MODE1(MSDC1_CLK) MODE2(LTE_MD32_JTAG_TCK) MODE3(C2K_TCK) MODE4(TDD_TCK) MODE5(CONN_DSP_JCK) MODE6(JTCK) MODE7(CONN_MCU_AICE_TCKC) PD SMT_GROUP(33) +GPIO168 = MODE0(GPIO168) MODE1(MSDC1_DAT0) MODE2(LTE_MD32_JTAG_TDI) MODE3(C2K_TDI) MODE4(TDD_TDI) MODE5(CONN_DSP_JDI) MODE6(JTDI) MODE7() PU SMT_GROUP(34) +GPIO169 = MODE0(GPIO169) MODE1(MSDC1_DAT1) MODE2(LTE_MD32_JTAG_TDO) MODE3(C2K_TDO) MODE4(TDD_TDO) MODE5(CONN_DSP_JDO) MODE6(JTDO) MODE7() PU SMT_GROUP(34) +GPIO170 = MODE0(GPIO170) MODE1(MSDC1_DAT2) MODE2(LTE_MD32_JTAG_TRST) MODE3(C2K_NTRST) MODE4(TDD_TRSTN) MODE5(CONN_DSP_JINTP) MODE6(DM_JTINTP) MODE7() PU SMT_GROUP(34) +GPIO171 = MODE0(GPIO171) MODE1(MSDC1_DAT3) MODE2() MODE3(C2K_RTCK) MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(34) +GPIO172 = MODE0(GPIO172) MODE1(MSDC0_CMD) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(35) +GPIO173 = MODE0(GPIO173) MODE1(MSDC0_DSL) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(36) +GPIO174 = MODE0(GPIO174) MODE1(MSDC0_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(37) +GPIO175 = MODE0(GPIO175) MODE1(MSDC0_DAT0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO176 = MODE0(GPIO176) MODE1(MSDC0_DAT1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO177 = MODE0(GPIO177) MODE1(MSDC0_DAT2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO178 = MODE0(GPIO178) MODE1(MSDC0_DAT3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO179 = MODE0(GPIO179) MODE1(MSDC0_DAT4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO180 = MODE0(GPIO180) MODE1(MSDC0_DAT5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO181 = MODE0(GPIO181) MODE1(MSDC0_DAT6) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO182 = MODE0(GPIO182) MODE1(MSDC0_DAT7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO183 = MODE0(GPIO183) MODE1(MSDC0_RSTB) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(39) +GPIO184 = MODE0(GPIO184) MODE1(F2W_DATA) MODE2(MRG_CLK) MODE3(C2K_DM_EINT2) MODE4(PCM0_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(40) +GPIO185 = MODE0(GPIO185) MODE1(F2W_CK) MODE2(MRG_DI) MODE3(C2K_DM_EINT3) MODE4(PCM0_DI) MODE5() MODE6() MODE7() PD SMT_GROUP(40) +GPIO186 = MODE0(GPIO186) MODE1(WB_RSTB) MODE2() MODE3() MODE4(URXD3) MODE5(UTXD3) MODE6() MODE7() PD SMT_GROUP(41) +GPIO187 = MODE0(GPIO187) MODE1(WB_SCLK) MODE2(MRG_DO) MODE3() MODE4(PCM0_DO) MODE5() MODE6() MODE7() PD SMT_GROUP(41) +GPIO188 = MODE0(GPIO188) MODE1(WB_SDATA) MODE2(MRG_SYNC) MODE3() MODE4(PCM0_SYNC) MODE5() MODE6() MODE7() PD SMT_GROUP(41) +GPIO189 = MODE0(GPIO189) MODE1(WB_SEN) MODE2() MODE3() MODE4(UTXD3) MODE5(URXD3) MODE6() MODE7() PD SMT_GROUP(41) +GPIO190 = MODE0(GPIO190) MODE1(GPS_RXQN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO191 = MODE0(GPIO191) MODE1(GPS_RXQP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO192 = MODE0(GPIO192) MODE1(GPS_RXIN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO193 = MODE0(GPIO193) MODE1(GPS_RXIP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO194 = MODE0(GPIO194) MODE1(WB_RXQN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO195 = MODE0(GPIO195) MODE1(WB_RXQP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO196 = MODE0(GPIO196) MODE1(WB_RXIN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO197 = MODE0(GPIO197) MODE1(WB_RXIP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- + +[GPO] + +[EINT] +EINT_COUNT = 213 +EINT_DEBOUNCE_TIME_COUNT = 12 + +[WAKE_UP_SRC] +ACCDET +MSDC1_INS +MSDC2_INS +CHR_STAT +IRQ_NFC +COMBO_BGF +WIFI +TOUCH_PANEL +ALS +MHL +EXT_BUCK_OC + +[EINT_MD1] +EINT_MD1_COUNT = 4 + +[ADC] +ADC_COUNT = 6 + +[ADC_EX_PIN] +0 +1 +12 +13 +14 +15 + +[KEYPAD] +KEY_ROW = 8 +KEY_COLUMN = 9 + +[KEYPAD_EXTEND_TYPE] +KEY_ROW = 3 +KEY_COLUMN = 6 +KEY_DESIGN_NOTICES = "Please notice that the circuit design are different between single and double keypad, and DO NOT add 1K ohm resistor on double keypad" + +[I2C] +I2C_COUNT = 32 +CHANNEL_COUNT = 4 + +[CLK_BUF] +CLK_BUF_COUNT = 4 + +[POWER] +DVDD28_SIM1 +DVDD28_SIM2 +DVDD28_MC1 +DVDD28_MC2 + +[DVDD28_SIM1] +VIO18 +VIO28 + +[DVDD28_SIM2] +VIO18 +VIO28 + +[DVDD28_MC1] +VIO18 +VIO28 + +[DVDD28_MC2] +VIO18 +VIO28 + +[MSDC_POWER_MC1] +MSDC_VIO18_MC1 +MSDC_VIO28_MC1 +MSDC_VMC + + +[SRC_PIN] +PAD_ANT_SEL1 +PAD_ANT_SEL2 +PAD_CMDAT1 +PAD_EINT1 +PAD_EINT2 +PAD_EINT3 +PAD_EINT8 +PAD_EINT9 +PAD_EINT10 +PAD_EINT11 +PAD_EINT12 +PAD_SCL3 +PAD_SDA3 + +[SRC_PIN_INDEX] +2 +1 +0 +0 +0 +0 +1 +2 +0 +1 +2 +1 +2 + diff --git a/tools/dct/MT6753.fig b/tools/dct/MT6753.fig new file mode 100755 index 000000000..f5d554ce8 --- /dev/null +++ b/tools/dct/MT6753.fig @@ -0,0 +1,333 @@ +[Chip Type] +Chip = MT6753 +GPIO_Pull_Sel = 1 +PMIC_Config = 1 +PMIC_ON_OFF_CONFIG = 1 +GPIO_ModeNum = 8 +EINT_MD1_Config = 1 +EINT_MD1_SRC_PIN = 1 +POWER_Config = 1 +POWER_COUNT = 4 +SpecialKey_Config = 1 +I2C_Config = 1 +CLOCK_BUFFER_CONFIG = 1 +Extend_Key_Config=1 +Reset_Key_Config = 1 +PMIC_APP_Ver = 2 +PMIC_APP_COUNT = 6 +MD1_EINT_SRC_PIN_CFG = 1 +AndroidPhone = 1 + +[GPIO] +GPIO0 = MODE0(GPIO0) MODE1(IDDIG) MODE2(DPI_D4) MODE3(CLKM4) MODE4(EXT_FRAME_SYNC) MODE5(PWM3) MODE6(KCOL2) MODE7(C2K_ARM_EINT0) PD SMT_GROUP(1) +GPIO1 = MODE0(GPIO1) MODE1(PWM2) MODE2(DPI_D5) MODE3(MD_EINT0) MODE4(TDD_TDO) MODE5(CONN_MCU_TDO) MODE6(PTA_RXD) MODE7(C2K_ARM_EINT1) PD SMT_GROUP(1) +GPIO2 = MODE0(GPIO2) MODE1(CLKM0) MODE2(DPI_D6) MODE3(MD_EINT0) MODE4(USB_DRVVBUS) MODE5(CONN_MCU_DBGACK_N) MODE6(PTA_TXD) MODE7(C2K_ARM_EINT2) PD SMT_GROUP(1) +GPIO3 = MODE0(GPIO3) MODE1(CLKM1) MODE2(DPI_D7) MODE3(SPI_MIB) MODE4(MD_EINT0) MODE5(CONN_MCU_DBGI_N) MODE6(CONN_MCU_AICE_TMSC) MODE7(C2K_ARM_EINT3) PD SMT_GROUP(1) +GPIO4 = MODE0(GPIO4) MODE1(CLKM2) MODE2(DPI_D8) MODE3(SPI_MOB) MODE4(TDD_TCK) MODE5(CONN_MCU_TCK[0]) MODE6(CONN_MCU_AICE_TCKC) MODE7(C2K_DM_EINT0) PD SMT_GROUP(1) +GPIO5 = MODE0(GPIO5) MODE1(UCTS2) MODE2(DPI_D9) MODE3(SPI_CSB) MODE4(TDD_TDI) MODE5(CONN_MCU_TDI) MODE6(I2S1_DO) MODE7(MD_URXD) PD SMT_GROUP(2) +GPIO6 = MODE0(GPIO6) MODE1(URTS2) MODE2(DPI_D10) MODE3(SPI_CKB) MODE4(TDD_TRSTN) MODE5(CONN_MCU_TRST_B) MODE6(I2S1_LRCK) MODE7(MD_UTXD) PD SMT_GROUP(2) +GPIO7 = MODE0(GPIO7) MODE1(UCTS3) MODE2(DPI_D11) MODE3(SDA1) MODE4(TDD_TMS) MODE5(CONN_MCU_TMS) MODE6(I2S1_BCK) MODE7(TDD_TXD) PD SMT_GROUP(2) +GPIO8 = MODE0(GPIO8) MODE1(URTS3) MODE2(C2K_UIM0_HOT_PLUG_IN) MODE3(SCL1) MODE4(PCM1_DO1) MODE5(MD_EINT1) MODE6(KCOL4) MODE7(UTXD0) PD SMT_GROUP(2) +GPIO9 = MODE0(GPIO9) MODE1(C2K_UIM1_HOT_PLUG_IN) MODE2(PCM1_DO0) MODE3(I2S3_MCK) MODE4(MD_EINT2) MODE5(CLKM2) MODE6(I2S1_MCK) MODE7(DBG_MON_A29) PD SMT_GROUP(3) +GPIO10 = MODE0(GPIO10) MODE1(PWM1) MODE2(CLKM1) MODE3(KROW2) MODE4(MD_EINT0) MODE5(I2S1_MCK) MODE6(SDA3) MODE7(DBG_MON_A30) PD SMT_GROUP(3) +GPIO11 = MODE0(GPIO11) MODE1(MD_EINT1) MODE2(IRTX_OUT) MODE3(C2K_UIM0_HOT_PLUG_IN) MODE4(CLKM0) MODE5(I2S2_MCK) MODE6(SCL3) MODE7(URXD0) PD SMT_GROUP(3) +GPIO12 = MODE0(GPIO12) MODE1(I2S0_MCK) MODE2(C2K_UIM1_HOT_PLUG_IN) MODE3(KCOL2) MODE4(MD_EINT2) MODE5(IRTX_OUT) MODE6(SRCLKENAI2) MODE7(PCM1_DO1) PD SMT_GROUP(3) +GPIO13 = MODE0(GPIO13) MODE1(WB_CTRL0) MODE2() MODE3(C2K_ARM_EINT0) MODE4() MODE5() MODE6() MODE7(DBG_MON_A0) PD SMT_GROUP(4) +GPIO14 = MODE0(GPIO14) MODE1(WB_CTRL1) MODE2() MODE3(C2K_ARM_EINT1) MODE4() MODE5() MODE6() MODE7(DBG_MON_A1) PD SMT_GROUP(4) +GPIO15 = MODE0(GPIO15) MODE1(WB_CTRL2) MODE2() MODE3(C2K_ARM_EINT2) MODE4() MODE5() MODE6() MODE7(DBG_MON_A2) PD SMT_GROUP(4) +GPIO16 = MODE0(GPIO16) MODE1(WB_CTRL3) MODE2() MODE3(C2K_ARM_EINT3) MODE4() MODE5() MODE6() MODE7(DBG_MON_A3) PD SMT_GROUP(4) +GPIO17 = MODE0(GPIO17) MODE1(WB_CTRL4) MODE2() MODE3(C2K_DM_EINT0) MODE4(WATCHDOG) MODE5() MODE6() MODE7(DBG_MON_A4) PD SMT_GROUP(4) +GPIO18 = MODE0(GPIO18) MODE1(WB_CTRL5) MODE2() MODE3(C2K_DM_EINT1) MODE4() MODE5() MODE6() MODE7(DBG_MON_A5) PD SMT_GROUP(4) +GPIO19 = MODE0(GPIO19) MODE1(ANT_SEL0) MODE2(IRTX_OUT) MODE3(IRDA_TX) MODE4(C2K_UART0_TXD) MODE5(GPS_FRAME_SYNC) MODE6(LTE_UTXD) MODE7(DBG_MON_A6) PD SMT_GROUP(5) +GPIO20 = MODE0(GPIO20) MODE1(ANT_SEL1) MODE2(C2K_UIM1_HOT_PLUG_IN) MODE3(IRDA_RX) MODE4(C2K_UART0_RXD) MODE5(MD_EINT2) MODE6(LTE_URXD) MODE7(DBG_MON_A7) PD SMT_GROUP(5) +GPIO21 = MODE0(GPIO21) MODE1(ANT_SEL2) MODE2(PWM2) MODE3(IRDA_PDN) MODE4(CORESONIC_SWCK) MODE5(MD_EINT1) MODE6(C2K_UIM0_HOT_PLUG_IN) MODE7(DBG_MON_A8) PD SMT_GROUP(5) +GPIO22 = MODE0(GPIO22) MODE1(RDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO23 = MODE0(GPIO23) MODE1(RDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO24 = MODE0(GPIO24) MODE1(RDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO25 = MODE0(GPIO25) MODE1(RDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO26 = MODE0(GPIO26) MODE1(RCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO27 = MODE0(GPIO27) MODE1(RCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO28 = MODE0(GPIO28) MODE1(RDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO29 = MODE0(GPIO29) MODE1(RDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO30 = MODE0(GPIO30) MODE1(RDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO31 = MODE0(GPIO31) MODE1(RDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO32 = MODE0(GPIO32) MODE1(RDN0_A) MODE2(CMHSYNC) MODE3(CMCSD0) MODE4() MODE5() MODE6() MODE7() -- +GPIO33 = MODE0(GPIO33) MODE1(RDP0_A) MODE2(CMVSYNC) MODE3(CMCSD1) MODE4() MODE5() MODE6() MODE7() -- +GPIO34 = MODE0(GPIO34) MODE1(RDN1_A) MODE2(CMDAT9) MODE3(CMCSD2) MODE4() MODE5() MODE6() MODE7() -- +GPIO35 = MODE0(GPIO35) MODE1(RDP1_A) MODE2(CMDAT8) MODE3(CMCSD3) MODE4() MODE5() MODE6() MODE7() -- +GPIO36 = MODE0(GPIO36) MODE1(RCN_A) MODE2(CMDAT7) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO37 = MODE0(GPIO37) MODE1(RCP_A) MODE2(CMDAT6) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO38 = MODE0(GPIO38) MODE1(RDN2_A) MODE2(CMDAT5) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO39 = MODE0(GPIO39) MODE1(RDP2_A) MODE2(CMDAT4) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO40 = MODE0(GPIO40) MODE1(RDN3_A) MODE2(CMDAT3) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO41 = MODE0(GPIO41) MODE1(RDP3_A) MODE2(CMDAT2) MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO42 = MODE0(GPIO42) MODE1(CMDAT0) MODE2(CMCSD0) MODE3(CMMCLK1) MODE4() MODE5(ANT_SEL5) MODE6(CLKM5) MODE7(DBG_MON_A9) PD SMT_GROUP(6) +GPIO43 = MODE0(GPIO43) MODE1(CMDAT1) MODE2(CMCSD1) MODE3(CMFLASH) MODE4(MD_EINT0) MODE5(CMMCLK1) MODE6(CLKM4) MODE7(DBG_MON_A10) PD SMT_GROUP(6) +GPIO44 = MODE0(GPIO44) MODE1(CMPCLK) MODE2(CMCSK) MODE3(CMCSD2) MODE4(KCOL3) MODE5(SRCLKENAI2) MODE6(PWM0) MODE7(DBG_MON_A11) PD SMT_GROUP(6) +GPIO45 = MODE0(GPIO45) MODE1(CMMCLK0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A12) PD SMT_GROUP(7) +GPIO46 = MODE0(GPIO46) MODE1(CMMCLK1) MODE2(IDDIG) MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(DM_JTINTP) MODE6(KCOL6) MODE7(DBG_MON_A13) PD SMT_GROUP(7) +GPIO47 = MODE0(GPIO47) MODE1(SDA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(8) +GPIO48 = MODE0(GPIO48) MODE1(SCL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(8) +GPIO49 = MODE0(GPIO49) MODE1(SDA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(9) +GPIO50 = MODE0(GPIO50) MODE1(SCL1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(9) +GPIO51 = MODE0(GPIO51) MODE1(SDA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(10) +GPIO52 = MODE0(GPIO52) MODE1(SCL2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(10) +GPIO53 = MODE0(GPIO53) MODE1(SDA3) MODE2() MODE3(IDDIG) MODE4() MODE5(MD_EINT2) MODE6(C2K_UIM1_HOT_PLUG_IN) MODE7() PU SMT_GROUP(11) +GPIO54 = MODE0(GPIO54) MODE1(SCL3) MODE2() MODE3(IDDIG) MODE4() MODE5(MD_EINT1) MODE6(C2K_UIM0_HOT_PLUG_IN) MODE7() PU SMT_GROUP(11) +GPIO55 = MODE0(GPIO55) MODE1(SRCLKENAI0) MODE2(PWM2) MODE3(CLKM5) MODE4(CORESONIC_SWD) MODE5(ANT_SEL6) MODE6(KROW5) MODE7(DISP_PWM) PD SMT_GROUP(12) +GPIO56 = MODE0(GPIO56) MODE1(SRCLKENA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(13) +GPIO57 = MODE0(GPIO57) MODE1(URXD2) MODE2(DPI_HSYNC0) MODE3(UTXD2) MODE4(MD_URXD) MODE5(SRCLKENAI1) MODE6(KROW4) MODE7(DBG_MON_A14) PD SMT_GROUP(14) +GPIO58 = MODE0(GPIO58) MODE1(UTXD2) MODE2(DPI_VSYNC0) MODE3(URXD2) MODE4(MD_UTXD) MODE5(TDD_TXD) MODE6(KROW5) MODE7(DBG_MON_A15) PD SMT_GROUP(14) +GPIO59 = MODE0(GPIO59) MODE1(URXD3) MODE2(DPI_CK0) MODE3(UTXD3) MODE4(UCTS2) MODE5(PWM3) MODE6(KROW6) MODE7(DBG_MON_A16) PD SMT_GROUP(14) +GPIO60 = MODE0(GPIO60) MODE1(UTXD3) MODE2(DPI_DE0) MODE3(URXD3) MODE4(URTS2) MODE5(PWM4) MODE6(KROW7) MODE7(DBG_MON_A17) PD SMT_GROUP(14) +GPIO61 = MODE0(GPIO61) MODE1(PCM1_CLK[0]) MODE2(DPI_D0) MODE3(I2S0_BCK) MODE4(KROW4) MODE5(ANT_SEL3) MODE6(IRTX_OUT) MODE7(DBG_MON_A18) PD SMT_GROUP(15) +GPIO62 = MODE0(GPIO62) MODE1(PCM1_SYNC) MODE2(DPI_D1) MODE3(I2S0_LRCK) MODE4(KCOL7) MODE5(CLKM3) MODE6(CMFLASH) MODE7(DBG_MON_A19) PD SMT_GROUP(15) +GPIO63 = MODE0(GPIO63) MODE1(PCM1_DI) MODE2(DPI_D2) MODE3(I2S0_DI) MODE4(PCM1_DO0) MODE5(CLKM5) MODE6(KROW3) MODE7(DBG_MON_A20) PD SMT_GROUP(15) +GPIO64 = MODE0(GPIO64) MODE1(PCM1_DO0) MODE2(DPI_D3) MODE3(I2S0_MCK) MODE4(PCM1_DI) MODE5(SRCLKENAI2) MODE6(KCOL5) MODE7(DBG_MON_A21) PD SMT_GROUP(15) +GPIO65 = MODE0(GPIO65) MODE1(SPI_CSA) MODE2(EXT_FRAME_SYNC) MODE3(I2S3_MCK) MODE4(KROW2) MODE5(GPS_FRAME_SYNC) MODE6(PTA_RXD) MODE7(DBG_MON_A22) PD SMT_GROUP(16) +GPIO66 = MODE0(GPIO66) MODE1(SPI_CKA) MODE2(USB_DRVVBUS) MODE3(I2S3_BCK) MODE4(KCOL2) MODE5() MODE6(PTA_TXD) MODE7(DBG_MON_A23) PD SMT_GROUP(16) +GPIO67 = MODE0(GPIO67) MODE1(SPI_MIA) MODE2(SPI_MOA) MODE3(I2S3_DO) MODE4(PTA_RXD) MODE5(IDDIG) MODE6(UCTS1) MODE7(DBG_MON_A24) PD SMT_GROUP(16) +GPIO68 = MODE0(GPIO68) MODE1(SPI_MOA) MODE2(SPI_MIA) MODE3(I2S3_LRCK) MODE4(PTA_TXD) MODE5(ANT_SEL4) MODE6(URTS1) MODE7(DBG_MON_A25) PD SMT_GROUP(16) +GPIO69 = MODE0(GPIO69) MODE1(DISP_PWM) MODE2(PWM1) MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(ANT_SEL7) MODE6(DM_JTINTP) MODE7() PD SMT_GROUP(17) +GPIO70 = MODE0(GPIO70) MODE1(JTMS) MODE2(CONN_MCU_TMS) MODE3(LTE_MD32_JTAG_TMS) MODE4(TDD_TMS) MODE5(CORESONIC_SWD) MODE6(DM_OTMS) MODE7(DFD_TMS) PU SMT_GROUP(18) +GPIO71 = MODE0(GPIO71) MODE1(JTCK) MODE2(CONN_MCU_TCK[1]) MODE3(LTE_MD32_JTAG_TCK) MODE4(TDD_TCK) MODE5(CORESONIC_SWCK) MODE6(DM_OTCK) MODE7(DFD_TCK_XI) PU SMT_GROUP(18) +GPIO72 = MODE0(GPIO72) MODE1(JTDI) MODE2(CONN_MCU_TDI) MODE3(LTE_MD32_JTAG_TDI) MODE4(TDD_TDI) MODE5() MODE6(DM_OTDI) MODE7(DFD_TDI) PU SMT_GROUP(18) +GPIO73 = MODE0(GPIO73) MODE1(JTDO) MODE2(CONN_MCU_TDO) MODE3(LTE_MD32_JTAG_TDO) MODE4(TDD_TDO) MODE5() MODE6(DM_OTDO) MODE7(DFD_TDO) PU SMT_GROUP(18) +GPIO74 = MODE0(GPIO74) MODE1(URXD0) MODE2(UTXD0) MODE3(MD_URXD) MODE4(SDA3) MODE5(C2K_UART0_RXD) MODE6(LTE_URXD) MODE7(AUXIF_ST) PU SMT_GROUP(19) +GPIO75 = MODE0(GPIO75) MODE1(UTXD0) MODE2(URXD0) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5(C2K_UART0_TXD) MODE6(LTE_UTXD) MODE7() PU SMT_GROUP(19) +GPIO76 = MODE0(GPIO76) MODE1(URXD1) MODE2(UTXD1) MODE3(MD_URXD) MODE4(SCL3) MODE5(LTE_URXD) MODE6(C2K_UART0_RXD) MODE7(AUXIF_CLK) PD SMT_GROUP(19) +GPIO77 = MODE0(GPIO77) MODE1(UTXD1) MODE2(URXD1) MODE3(MD_UTXD) MODE4(TDD_TXD) MODE5(LTE_UTXD) MODE6(C2K_UART0_TXD) MODE7() PD SMT_GROUP(19) +GPIO78 = MODE0(GPIO78) MODE1(I2S0_DI) MODE2(PCM1_DI) MODE3(I2S3_DO) MODE4(I2S1_DO) MODE5(PWM0) MODE6(I2S2_DI) MODE7(DBG_MON_A26) PD SMT_GROUP(20) +GPIO79 = MODE0(GPIO79) MODE1(I2S0_LRCK) MODE2(PCM1_SYNC) MODE3(I2S3_LRCK) MODE4(I2S1_LRCK) MODE5(PWM3) MODE6(I2S2_LRCK) MODE7(DBG_MON_A27) PD SMT_GROUP(20) +GPIO80 = MODE0(GPIO80) MODE1(I2S0_BCK) MODE2(PCM1_CLK[1]) MODE3(I2S3_BCK) MODE4(I2S1_BCK) MODE5(PWM4) MODE6(I2S2_BCK) MODE7(DBG_MON_A28) PD SMT_GROUP(20) +GPIO81 = MODE0(GPIO81) MODE1(KROW0) MODE2() MODE3(CONN_MCU_DBGI_N) MODE4(CORESONIC_SWCK) MODE5(C2K_TCK) MODE6() MODE7(C2K_DM_EINT1) PD SMT_GROUP(21) +GPIO82 = MODE0(GPIO82) MODE1(KROW1) MODE2() MODE3(CONN_MCU_TRST_B) MODE4(CORESONIC_SWD) MODE5(C2K_NTRST) MODE6(USB_DRVVBUS) MODE7(C2K_DM_EINT2) PD SMT_GROUP(21) +GPIO83 = MODE0(GPIO83) MODE1(KROW2) MODE2(USB_DRVVBUS) MODE3() MODE4() MODE5(C2K_TDI) MODE6() MODE7(C2K_DM_EINT3) PD SMT_GROUP(21) +GPIO84 = MODE0(GPIO84) MODE1(KCOL0) MODE2(URTS0) MODE3(CONN_MCU_DBGACK_N) MODE4(SCL2) MODE5(C2K_TDO) MODE6(AUXIF_CLK) MODE7() PU SMT_GROUP(21) +GPIO85 = MODE0(GPIO85) MODE1(KCOL1) MODE2(UCTS0) MODE3(UCTS1) MODE4(SDA2) MODE5(C2K_TMS) MODE6(AUXIF_ST) MODE7(DBG_MON_A31) PD SMT_GROUP(21) +GPIO86 = MODE0(GPIO86) MODE1(KCOL2) MODE2() MODE3(URTS1) MODE4() MODE5(C2K_RTCK) MODE6() MODE7(DBG_MON_A32) PD SMT_GROUP(21) +GPIO87 = MODE0(GPIO87) MODE1(BPI_BUS5) MODE2(LTE_C2K_BPI_BUS5) MODE3() MODE4() MODE5(C2K_BPI_BUS5) MODE6() MODE7(DBG_MON_B0) PD SMT_GROUP(22) +GPIO88 = MODE0(GPIO88) MODE1(BPI_BUS6) MODE2(LTE_C2K_BPI_BUS6) MODE3() MODE4() MODE5(C2K_BPI_BUS6) MODE6() MODE7(DBG_MON_B1) PD SMT_GROUP(22) +GPIO89 = MODE0(GPIO89) MODE1(BPI_BUS7) MODE2(LTE_C2K_BPI_BUS7) MODE3(CLKM0) MODE4() MODE5(C2K_BPI_BUS7) MODE6() MODE7(DBG_MON_B2) PD SMT_GROUP(22) +GPIO90 = MODE0(GPIO90) MODE1(BPI_BUS8) MODE2(LTE_C2K_BPI_BUS8) MODE3(CLKM1) MODE4() MODE5(C2K_BPI_BUS8) MODE6() MODE7(DBG_MON_B3) PD SMT_GROUP(22) +GPIO91 = MODE0(GPIO91) MODE1(BPI_BUS9) MODE2(LTE_C2K_BPI_BUS9) MODE3(CLKM2) MODE4() MODE5(C2K_BPI_BUS9) MODE6() MODE7(DBG_MON_B4) PD SMT_GROUP(22) +GPIO92 = MODE0(GPIO92) MODE1(BPI_BUS10) MODE2(LTE_C2K_BPI_BUS10) MODE3(CLKM3) MODE4() MODE5(C2K_BPI_BUS10) MODE6() MODE7(DBG_MON_B5) PD SMT_GROUP(22) +GPIO93 = MODE0(GPIO93) MODE1(BPI_BUS11) MODE2(LTE_C2K_BPI_BUS11) MODE3() MODE4() MODE5(C2K_BPI_BUS11) MODE6() MODE7(DBG_MON_B6) PD SMT_GROUP(22) +GPIO94 = MODE0(GPIO94) MODE1(BPI_BUS12) MODE2(LTE_C2K_BPI_BUS12) MODE3() MODE4() MODE5(C2K_BPI_BUS12) MODE6() MODE7(DBG_MON_B7) PD SMT_GROUP(22) +GPIO95 = MODE0(GPIO95) MODE1(BPI_BUS13) MODE2(LTE_C2K_BPI_BUS13) MODE3() MODE4() MODE5(C2K_BPI_BUS13) MODE6() MODE7(DBG_MON_B8) PD SMT_GROUP(22) +GPIO96 = MODE0(GPIO96) MODE1(BPI_BUS14) MODE2(LTE_C2K_BPI_BUS14) MODE3() MODE4() MODE5(C2K_BPI_BUS14) MODE6() MODE7(DBG_MON_B9) PD SMT_GROUP(22) +GPIO97 = MODE0(GPIO97) MODE1(BPI_BUS15) MODE2(LTE_C2K_BPI_BUS15) MODE3() MODE4() MODE5(C2K_BPI_BUS15) MODE6() MODE7(DBG_MON_B10) PD SMT_GROUP(22) +GPIO98 = MODE0(GPIO98) MODE1(BPI_BUS16) MODE2(LTE_C2K_BPI_BUS16) MODE3() MODE4() MODE5(C2K_BPI_BUS16) MODE6() MODE7(DBG_MON_B11) PD SMT_GROUP(22) +GPIO99 = MODE0(GPIO99) MODE1(BPI_BUS17) MODE2(LTE_C2K_BPI_BUS17) MODE3() MODE4(I2S0_DI) MODE5(C2K_BPI_BUS17) MODE6() MODE7(DBG_MON_B12) PD SMT_GROUP(22) +GPIO100 = MODE0(GPIO100) MODE1(BPI_BUS18) MODE2(LTE_C2K_BPI_BUS18) MODE3() MODE4() MODE5(C2K_BPI_BUS18) MODE6() MODE7(DBG_MON_B13) PD SMT_GROUP(22) +GPIO101 = MODE0(GPIO101) MODE1(BPI_BUS19) MODE2(LTE_C2K_BPI_BUS19) MODE3() MODE4() MODE5(C2K_BPI_BUS19) MODE6() MODE7(DBG_MON_B14) PD SMT_GROUP(22) +GPIO102 = MODE0(GPIO102) MODE1(BPI_BUS20) MODE2(LTE_C2K_BPI_BUS20) MODE3() MODE4() MODE5(C2K_BPI_BUS20) MODE6() MODE7(DBG_MON_B15) PD SMT_GROUP(22) +GPIO103 = MODE0(GPIO103) MODE1(C2K_TXBPI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B16) PD SMT_GROUP(22) +GPIO104 = MODE0(GPIO104) MODE1(RFIC1_BSI_EN) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_EN) MODE6() MODE7(DBG_MON_B17) PD SMT_GROUP(23) +GPIO105 = MODE0(GPIO105) MODE1(RFIC1_BSI_CK) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_CLK) MODE6() MODE7(DBG_MON_B18) PD SMT_GROUP(23) +GPIO106 = MODE0(GPIO106) MODE1(RFIC1_BSI_D0) MODE2() MODE3() MODE4() MODE5(C2K_RX_BSI_DATA) MODE6() MODE7(DBG_MON_B19) PD SMT_GROUP(23) +GPIO107 = MODE0(GPIO107) MODE1(RFIC1_BSI_D1) MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_EN) MODE6() MODE7(DBG_MON_B20) PD SMT_GROUP(23) +GPIO108 = MODE0(GPIO108) MODE1(RFIC1_BSI_D2) MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_CLK) MODE6() MODE7(DBG_MON_B21) PD SMT_GROUP(23) +GPIO109 = MODE0(GPIO109) MODE1() MODE2() MODE3() MODE4() MODE5(C2K_TX_BSI_DATA) MODE6() MODE7(DBG_MON_B22) PD SMT_GROUP(23) +GPIO110 = MODE0(GPIO110) MODE1(RFIC0_BSI_EN) MODE2() MODE3() MODE4(SPM_BSI_EN) MODE5() MODE6() MODE7(DBG_MON_B23) PD SMT_GROUP(23) +GPIO111 = MODE0(GPIO111) MODE1(RFIC0_BSI_CK) MODE2() MODE3() MODE4(SPM_BSI_CLK) MODE5() MODE6() MODE7(DBG_MON_B24) PD SMT_GROUP(23) +GPIO112 = MODE0(GPIO112) MODE1(RFIC0_BSI_D2) MODE2() MODE3() MODE4(SPM_BSI_D2) MODE5() MODE6() MODE7(DBG_MON_B25) PD SMT_GROUP(23) +GPIO113 = MODE0(GPIO113) MODE1(RFIC0_BSI_D1) MODE2() MODE3() MODE4(SPM_BSI_D1) MODE5() MODE6() MODE7(DBG_MON_B26) PD SMT_GROUP(23) +GPIO114 = MODE0(GPIO114) MODE1(RFIC0_BSI_D0) MODE2() MODE3() MODE4(SPM_BSI_D0) MODE5() MODE6() MODE7(DBG_MON_B27) PD SMT_GROUP(23) +GPIO115 = MODE0(GPIO115) MODE1(AUXIN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO116 = MODE0(GPIO116) MODE1(AUXIN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO117 = MODE0(GPIO117) MODE1(AUXIN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO118 = MODE0(GPIO118) MODE1(TXBPI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO119 = MODE0(GPIO119) MODE1(BPI_BUS0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B28) PD SMT_GROUP(24) +GPIO120 = MODE0(GPIO120) MODE1(BPI_BUS1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B29) PD SMT_GROUP(24) +GPIO121 = MODE0(GPIO121) MODE1(BPI_BUS2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B30) PD SMT_GROUP(24) +GPIO122 = MODE0(GPIO122) MODE1(BPI_BUS3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B31) PD SMT_GROUP(24) +GPIO123 = MODE0(GPIO123) MODE1(BPI_BUS4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B32) PD SMT_GROUP(24) +GPIO124 = MODE0(GPIO124) MODE1(BPI_BUS21) MODE2(SDA_EXTB) MODE3() MODE4() MODE5(DPI_HSYNC1) MODE6(KCOL2) MODE7(TDD_TXD) PD SMT_GROUP(24) +GPIO125 = MODE0(GPIO125) MODE1(BPI_BUS22) MODE2(SCL_EXTB) MODE3() MODE4() MODE5(DPI_VSYNC1) MODE6(KROW2) MODE7(MD_URXD) PD SMT_GROUP(24) +GPIO126 = MODE0(GPIO126) MODE1(BPI_BUS23) MODE2(SDA_EXTB) MODE3() MODE4() MODE5(DPI_CK1) MODE6(I2S2_MCK) MODE7(MD_UTXD) PD SMT_GROUP(24) +GPIO127 = MODE0(GPIO127) MODE1(BPI_BUS24) MODE2(SCL_EXTB) MODE3(CONN_MCU_DBGI_N) MODE4(EXT_FRAME_SYNC) MODE5(DPI_DE1) MODE6(SRCLKENAI1) MODE7(URXD0) PD SMT_GROUP(24) +GPIO128 = MODE0(GPIO128) MODE1(BPI_BUS25) MODE2() MODE3(GPS_FRAME_SYNC) MODE4(SDA_EXTB) MODE5(I2S2_DI) MODE6(PTA_RXD) MODE7(UTXD0) PD SMT_GROUP(24) +GPIO129 = MODE0(GPIO129) MODE1(BPI_BUS26) MODE2(DISP_PWM) MODE3() MODE4(SCL_EXTB) MODE5(I2S2_LRCK) MODE6(PTA_TXD) MODE7(LTE_URXD) PD SMT_GROUP(24) +GPIO130 = MODE0(GPIO130) MODE1(BPI_BUS27) MODE2() MODE3() MODE4() MODE5(I2S2_BCK) MODE6(IRTX_OUT) MODE7(LTE_UTXD) PD SMT_GROUP(24) +GPIO131 = MODE0(GPIO131) MODE1(LTE_PAVM0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO132 = MODE0(GPIO132) MODE1(LTE_PAVM1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO133 = MODE0(GPIO133) MODE1(MIPI1_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO134 = MODE0(GPIO134) MODE1(MIPI1_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO135 = MODE0(GPIO135) MODE1(MIPI0_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO136 = MODE0(GPIO136) MODE1(MIPI0_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(24) +GPIO137 = MODE0(GPIO137) MODE1(RTC32K_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(25) +GPIO138 = MODE0(GPIO138) MODE1(PWRAP_SPIDO) MODE2(PWRAP_SPIDI) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO139 = MODE0(GPIO139) MODE1(PWRAP_SPIDI) MODE2(PWRAP_SPIDO) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO140 = MODE0(GPIO140) MODE1() MODE2() MODE3(LTE_MD32_JTAG_TRST) MODE4(TDD_TRSTN) MODE5(DM_JTINTP) MODE6() MODE7() PD SMT_GROUP(26) +GPIO141 = MODE0(GPIO141) MODE1(PWRAP_SPICK_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(26) +GPIO142 = MODE0(GPIO142) MODE1(PWRAP_SPICS_B_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(26) +GPIO143 = MODE0(GPIO143) MODE1(AUD_CLK_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO144 = MODE0(GPIO144) MODE1(AUD_DAT_MISO) MODE2() MODE3(AUD_DAT_MOSI) MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO145 = MODE0(GPIO145) MODE1(AUD_DAT_MOSI) MODE2() MODE3(AUD_DAT_MISO) MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(27) +GPIO146 = MODE0(GPIO146) MODE1(LCM_RST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(28) +GPIO147 = MODE0(GPIO147) MODE1(DSI_TE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(28) +GPIO148 = MODE0(GPIO148) MODE1(SRCLKENA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(29) +GPIO149 = MODE0(GPIO149) MODE1(WATCHDOG) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(25) +GPIO150 = MODE0(GPIO150) MODE1(TDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO151 = MODE0(GPIO151) MODE1(TDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO152 = MODE0(GPIO152) MODE1(TDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO153 = MODE0(GPIO153) MODE1(TDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO154 = MODE0(GPIO154) MODE1(TCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO155 = MODE0(GPIO155) MODE1(TCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO156 = MODE0(GPIO156) MODE1(TDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO157 = MODE0(GPIO157) MODE1(TDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO158 = MODE0(GPIO158) MODE1(TDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO159 = MODE0(GPIO159) MODE1(TDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO160 = MODE0(GPIO160) MODE1(MD_SIM2_SCLK) MODE2(MD_SIM1_SCLK) MODE3(UIM0_CLK) MODE4(UIM1_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO161 = MODE0(GPIO161) MODE1(MD_SIM2_SRST) MODE2(MD_SIM1_SRST) MODE3(UIM0_RST) MODE4(UIM1_RST) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO162 = MODE0(GPIO162) MODE1(MD_SIM2_SDAT) MODE2(MD_SIM1_SDAT) MODE3(UIM0_IO) MODE4(UIM1_IO) MODE5() MODE6() MODE7() PD SMT_GROUP(30) +GPIO163 = MODE0(GPIO163) MODE1(MD_SIM1_SCLK) MODE2(MD_SIM2_SCLK) MODE3(UIM1_CLK) MODE4(UIM0_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO164 = MODE0(GPIO164) MODE1(MD_SIM1_SRST) MODE2(MD_SIM2_SRST) MODE3(UIM1_RST) MODE4(UIM0_RST) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO165 = MODE0(GPIO165) MODE1(MD_SIM1_SDAT) MODE2(MD_SIM2_SDAT) MODE3(UIM1_IO) MODE4(UIM0_IO) MODE5() MODE6() MODE7() PD SMT_GROUP(31) +GPIO166 = MODE0(GPIO166) MODE1(MSDC1_CMD) MODE2(LTE_MD32_JTAG_TMS) MODE3(C2K_TMS) MODE4(TDD_TMS) MODE5(CONN_DSP_JMS) MODE6(JTMS) MODE7(CONN_MCU_AICE_TMSC) PU SMT_GROUP(32) +GPIO167 = MODE0(GPIO167) MODE1(MSDC1_CLK) MODE2(LTE_MD32_JTAG_TCK) MODE3(C2K_TCK) MODE4(TDD_TCK) MODE5(CONN_DSP_JCK) MODE6(JTCK) MODE7(CONN_MCU_AICE_TCKC) PD SMT_GROUP(33) +GPIO168 = MODE0(GPIO168) MODE1(MSDC1_DAT0) MODE2(LTE_MD32_JTAG_TDI) MODE3(C2K_TDI) MODE4(TDD_TDI) MODE5(CONN_DSP_JDI) MODE6(JTDI) MODE7() PU SMT_GROUP(34) +GPIO169 = MODE0(GPIO169) MODE1(MSDC1_DAT1) MODE2(LTE_MD32_JTAG_TDO) MODE3(C2K_TDO) MODE4(TDD_TDO) MODE5(CONN_DSP_JDO) MODE6(JTDO) MODE7() PU SMT_GROUP(34) +GPIO170 = MODE0(GPIO170) MODE1(MSDC1_DAT2) MODE2(LTE_MD32_JTAG_TRST) MODE3(C2K_NTRST) MODE4(TDD_TRSTN) MODE5(CONN_DSP_JINTP) MODE6(DM_JTINTP) MODE7() PU SMT_GROUP(34) +GPIO171 = MODE0(GPIO171) MODE1(MSDC1_DAT3) MODE2() MODE3(C2K_RTCK) MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(34) +GPIO172 = MODE0(GPIO172) MODE1(MSDC0_CMD) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(35) +GPIO173 = MODE0(GPIO173) MODE1(MSDC0_DSL) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(36) +GPIO174 = MODE0(GPIO174) MODE1(MSDC0_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(37) +GPIO175 = MODE0(GPIO175) MODE1(MSDC0_DAT0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO176 = MODE0(GPIO176) MODE1(MSDC0_DAT1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO177 = MODE0(GPIO177) MODE1(MSDC0_DAT2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO178 = MODE0(GPIO178) MODE1(MSDC0_DAT3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO179 = MODE0(GPIO179) MODE1(MSDC0_DAT4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO180 = MODE0(GPIO180) MODE1(MSDC0_DAT5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO181 = MODE0(GPIO181) MODE1(MSDC0_DAT6) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO182 = MODE0(GPIO182) MODE1(MSDC0_DAT7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(38) +GPIO183 = MODE0(GPIO183) MODE1(MSDC0_RSTB) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(39) +GPIO184 = MODE0(GPIO184) MODE1(F2W_DATA) MODE2(MRG_CLK) MODE3(C2K_DM_EINT2) MODE4(PCM0_CLK) MODE5() MODE6() MODE7() PD SMT_GROUP(40) +GPIO185 = MODE0(GPIO185) MODE1(F2W_CK) MODE2(MRG_DI) MODE3(C2K_DM_EINT3) MODE4(PCM0_DI) MODE5() MODE6() MODE7() PD SMT_GROUP(40) +GPIO186 = MODE0(GPIO186) MODE1(WB_RSTB) MODE2() MODE3() MODE4(URXD3) MODE5(UTXD3) MODE6() MODE7() PD SMT_GROUP(41) +GPIO187 = MODE0(GPIO187) MODE1(WB_SCLK) MODE2(MRG_DO) MODE3() MODE4(PCM0_DO) MODE5() MODE6() MODE7() PD SMT_GROUP(41) +GPIO188 = MODE0(GPIO188) MODE1(WB_SDATA) MODE2(MRG_SYNC) MODE3() MODE4(PCM0_SYNC) MODE5() MODE6() MODE7() PD SMT_GROUP(41) +GPIO189 = MODE0(GPIO189) MODE1(WB_SEN) MODE2() MODE3() MODE4(UTXD3) MODE5(URXD3) MODE6() MODE7() PD SMT_GROUP(41) +GPIO190 = MODE0(GPIO190) MODE1(GPS_RXQN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO191 = MODE0(GPIO191) MODE1(GPS_RXQP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO192 = MODE0(GPIO192) MODE1(GPS_RXIN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO193 = MODE0(GPIO193) MODE1(GPS_RXIP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO194 = MODE0(GPIO194) MODE1(WB_RXQN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO195 = MODE0(GPIO195) MODE1(WB_RXQP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO196 = MODE0(GPIO196) MODE1(WB_RXIN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO197 = MODE0(GPIO197) MODE1(WB_RXIP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO198 = MODE0(GPIO198) MODE1(MSDC2_CMD) MODE2(SDA1) MODE3(C2K_UART0_RXD) MODE4(C2K_TMS) MODE5(ANT_SEL6) MODE6() MODE7(DM_OTMS) PD SMT_GROUP(42) +GPIO199 = MODE0(GPIO199) MODE1(MSDC2_CLK) MODE2(SCL1) MODE3(C2K_UART0_TXD) MODE4(C2K_TCK) MODE5(ANT_SEL7) MODE6(TDD_TXD) MODE7(DM_OTCK) PD SMT_GROUP(43) +GPIO200 = MODE0(GPIO200) MODE1(MSDC2_DAT0) MODE2(ANT_SEL6) MODE3(GPS_FRAME_SYNC) MODE4(C2K_TDI) MODE5(UTXD0) MODE6() MODE7(DM_OTDI) PD SMT_GROUP(44) +GPIO201 = MODE0(GPIO201) MODE1(MSDC2_DAT1) MODE2(ANT_SEL3) MODE3(PWM0) MODE4(C2K_TDO) MODE5(URXD0) MODE6() MODE7(DM_OTDO) PD SMT_GROUP(44) +GPIO202 = MODE0(GPIO202) MODE1(MSDC2_DAT2) MODE2(ANT_SEL4) MODE3(SDA2) MODE4(C2K_NTRST) MODE5(UTXD1) MODE6(KCOL3) MODE7(DM_JTINTP) PD SMT_GROUP(44) +GPIO203 = MODE0(GPIO203) MODE1(MSDC2_DAT3) MODE2(ANT_SEL5) MODE3(SCL2) MODE4(C2K_RTCK) MODE5(URXD1) MODE6(KCOL6) MODE7() PD SMT_GROUP(44) + +[GPO] + +[EINT] +EINT_COUNT = 213 +EINT_DEBOUNCE_TIME_COUNT = 12 + +[WAKE_UP_SRC] +ACCDET +MSDC1_INS +MSDC2_INS +CHR_STAT +IRQ_NFC +COMBO_BGF +WIFI +TOUCH_PANEL +ALS +MHL +EXT_BUCK_OC + +[EINT_MD1] +EINT_MD1_COUNT = 4 + +[ADC] +ADC_COUNT = 6 + +[ADC_EX_PIN] +0 +1 +12 +13 +14 +15 + +[KEYPAD] +KEY_ROW = 8 +KEY_COLUMN = 9 + +[KEYPAD_EXTEND_TYPE] +KEY_ROW = 3 +KEY_COLUMN = 6 +KEY_DESIGN_NOTICES = "Please notice that the circuit design are different between single and double keypad, and DO NOT add 1K ohm resistor on double keypad" + +[I2C] +I2C_COUNT = 32 +CHANNEL_COUNT = 5 + +[CLK_BUF] +CLK_BUF_COUNT = 4 + +[POWER] +DVDD28_SIM1 +DVDD28_SIM2 +DVDD28_MC1 +DVDD28_MC2 + +[DVDD28_SIM1] +VIO18 +VIO28 + +[DVDD28_SIM2] +VIO18 +VIO28 + +[DVDD28_MC1] +VIO18 +VIO28 + +[DVDD28_MC2] +VIO18 +VIO28 + +[MSDC_POWER_MC1] +MSDC_VIO18_MC1 +MSDC_VIO28_MC1 +MSDC_VMC + + +[SRC_PIN] +PAD_ANT_SEL1 +PAD_ANT_SEL2 +PAD_CMDAT1 +PAD_EINT1 +PAD_EINT2 +PAD_EINT3 +PAD_EINT8 +PAD_EINT9 +PAD_EINT10 +PAD_EINT11 +PAD_EINT12 +PAD_SCL3 +PAD_SDA3 + +[SRC_PIN_INDEX] +2 +1 +0 +0 +0 +0 +1 +2 +0 +1 +2 +1 +2 + diff --git a/tools/dct/MT8173.fig b/tools/dct/MT8173.fig new file mode 100644 index 000000000..b25e276ff --- /dev/null +++ b/tools/dct/MT8173.fig @@ -0,0 +1,453 @@ +[Chip Type] +Chip = MT8173 +GPIO_Pull_Sel = 1 +PMIC_Config = 1 +PMIC_GPIO_Config = 1 +PMIC_ON_OFF_CONFIG = 1 +EINT_EXT_Config = 1 +POWER_Config = 1 +POWER_COUNT = 55 +GPIO_ModeNum = 8 +AndroidPhone = 1 +SpecialKey_Config = 1 +I2C_Config = 1 +CLOCK_BUFFER_CONFIG = 1 +PMIC_APP_Ver = 2 +PMIC_APP_COUNT = 6 + +[GPIO] +GPIO0 = MODE0(GPIO0) MODE1(IRDA_PDN) MODE2(I2S1_WS) MODE3(AUD_SPDIF) MODE4(UTXD0) MODE5() MODE6() MODE7(DBG_MON_A_20_) PD SMT_GROUP(1) - +GPIO1 = MODE0(GPIO1) MODE1(IRDA_RXD) MODE2(I2S1_BCK) MODE3(SDA5) MODE4(URXD0) MODE5() MODE6() MODE7(DBG_MON_A_21_) PD SMT_GROUP(1) - +GPIO2 = MODE0(GPIO2) MODE1(IRDA_TXD) MODE2(I2S1_MCK) MODE3(SCL5) MODE4(UTXD3) MODE5() MODE6() MODE7(DBG_MON_A_22_) PD SMT_GROUP(1) - +GPIO3 = MODE0(GPIO3) MODE1(DSI1_TE) MODE2(I2S1_DO_1) MODE3(SDA3) MODE4(URXD3) MODE5() MODE6() MODE7(DBG_MON_A_23_) PD SMT_GROUP(1) - +GPIO4 = MODE0(GPIO4) MODE1(DISP_PWM1) MODE2(I2S1_DO_2) MODE3(SCL3) MODE4(UCTS3) MODE5() MODE6(SFWP_B) MODE7() PD SMT_GROUP(1) - +GPIO5 = MODE0(GPIO5) MODE1(PCM1_CLK) MODE2(I2S2_WS) MODE3(SPI_CK_3_) MODE4(URTS3) MODE5(AP_MD32_JTAG_TMS) MODE6(SFOUT) MODE7() PD SMT_GROUP(2) - +GPIO6 = MODE0(GPIO6) MODE1(PCM1_SYNC) MODE2(I2S2_BCK) MODE3(SPI_MI_3_) MODE4() MODE5(AP_MD32_JTAG_TCK) MODE6(SFCS0) MODE7() PU SMT_GROUP(2) - +GPIO7 = MODE0(GPIO7) MODE1(PCM1_DI) MODE2(I2S2_DI_1) MODE3(SPI_MO_3_) MODE4() MODE5(AP_MD32_JTAG_TDI) MODE6(SFHOLD) MODE7() PD SMT_GROUP(2) - +GPIO8 = MODE0(GPIO8) MODE1(PCM1_DO) MODE2(I2S2_DI_2) MODE3(SPI_CS_3_) MODE4(AUD_SPDIF) MODE5(AP_MD32_JTAG_TDO) MODE6(SFIN) MODE7() PD SMT_GROUP(2) - +GPIO9 = MODE0(GPIO9) MODE1(USB_DRVVBUS_P0) MODE2(I2S2_MCK) MODE3() MODE4(USB_DRVVBUS_P1) MODE5(AP_MD32_JTAG_TRST) MODE6(SFCK) MODE7() PD SMT_GROUP(2) - +GPIO10 = MODE0(GPIO10) MODE1(CLKM0) MODE2(DSI1_TE) MODE3(DISP_PWM1) MODE4(PWM4) MODE5(IRDA_RXD) MODE6() MODE7() PD SMT_GROUP(10) - +GPIO11 = MODE0(GPIO11) MODE1(CLKM1) MODE2(I2S3_WS) MODE3(USB_DRVVBUS_P0) MODE4(PWM5) MODE5(IRDA_TXD) MODE6(USB_DRVVBUS_P1) MODE7(DBG_MON_B_30_) PD SMT_GROUP(10) - +GPIO12 = MODE0(GPIO12) MODE1(CLKM2) MODE2(I2S3_BCK) MODE3(SRCLKENA0) MODE4() MODE5(I2S2_WS) MODE6() MODE7(DBG_MON_B_32_) PD SMT_GROUP(10) - +GPIO13 = MODE0(GPIO13) MODE1(CLKM3) MODE2(I2S3_MCK) MODE3(SRCLKENA0) MODE4() MODE5(I2S2_BCK) MODE6() MODE7(DBG_MON_A_32_) PD SMT_GROUP(10) - +GPIO14 = MODE0(GPIO14) MODE1(CMDAT0) MODE2(CMCSD0) MODE3() MODE4(CLKM2) MODE5() MODE6() MODE7(DBG_MON_B_6_) PD SMT_GROUP(26) - +GPIO15 = MODE0(GPIO15) MODE1(CMDAT1) MODE2(CMCSD1) MODE3(CMFLASH) MODE4(CLKM3) MODE5() MODE6() MODE7(DBG_MON_B_29_) PD SMT_GROUP(26) - +GPIO16 = MODE0(GPIO16) MODE1(IDDIG) MODE2(CMFLASH) MODE3() MODE4(PWM5) MODE5() MODE6() MODE7() PU SMT_GROUP(0) - +GPIO17 = MODE0(GPIO17) MODE1(WATCHDOG_AO) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(34) - +GPIO18 = MODE0(GPIO18) MODE1(CEC) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(19) open-drain +GPIO19 = MODE0(GPIO19) MODE1(HDMISCK) MODE2(HDCP_SCL) MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(19) open-drain +GPIO20 = MODE0(GPIO20) MODE1(HDMISD) MODE2(HDCP_SDA) MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(19) open-drain +GPIO21 = MODE0(GPIO21) MODE1(HTPLG) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(19) open-drain +GPIO22 = MODE0(GPIO22) MODE1(MSDC3_DAT0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU msdc3da_smt - +GPIO23 = MODE0(GPIO23) MODE1(MSDC3_DAT1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU msdc3da_smt - +GPIO24 = MODE0(GPIO24) MODE1(MSDC3_DAT2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU msdc3da_smt - +GPIO25 = MODE0(GPIO25) MODE1(MSDC3_DAT3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU msdc3da_smt - +GPIO26 = MODE0(GPIO26) MODE1(MSDC3_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD msdc3ck_smt - +GPIO27 = MODE0(GPIO27) MODE1(MSDC3_CMD) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU msdc3cm_smt - +GPIO28 = MODE0(GPIO28) MODE1(MSDC3_DSL) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD msdc3ds_smt - +GPIO29 = MODE0(GPIO29) MODE1(UCTS2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(3) - +GPIO30 = MODE0(GPIO30) MODE1(URTS2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(3) - +GPIO31 = MODE0(GPIO31) MODE1(URXD2) MODE2(UTXD2) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(3) - +GPIO32 = MODE0(GPIO32) MODE1(UTXD2) MODE2(URXD2) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(3) - +GPIO33 = MODE0(GPIO33) MODE1(MRG_CLK) MODE2(PCM0_CLK) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(4) - +GPIO34 = MODE0(GPIO34) MODE1(MRG_DI) MODE2(PCM0_DI) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(5) - +GPIO35 = MODE0(GPIO35) MODE1(MRG_DO) MODE2(PCM0_DO) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(5) - +GPIO36 = MODE0(GPIO36) MODE1(MRG_SYNC) MODE2(PCM0_SYNC) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(5) - +GPIO37 = MODE0(GPIO37) MODE1(USB_DRVVBUS_P0) MODE2(USB_DRVVBUS_P1) MODE3(PWM0) MODE4(PWM1) MODE5(PWM2) MODE6(CLKM0) MODE7() PD SMT_GROUP(6) - +GPIO38 = MODE0(GPIO38) MODE1(USB_DRVVBUS_P0) MODE2(USB_DRVVBUS_P1) MODE3() MODE4() MODE5() MODE6(CLKM1) MODE7() PD SMT_GROUP(6) - +GPIO39 = MODE0(GPIO39) MODE1(CM2MCLK) MODE2(CMCSD0) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_17_) PD SMT_GROUP(7) - +GPIO40 = MODE0(GPIO40) MODE1(CMPCLK) MODE2(CMCSK) MODE3(CMCSD2) MODE4() MODE5() MODE6() MODE7(DBG_MON_A_18_) PD SMT_GROUP(9) - +GPIO41 = MODE0(GPIO41) MODE1(CMMCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_19_) PD SMT_GROUP(9) - +GPIO42 = MODE0(GPIO42) MODE1(DSI_TE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(16) - +GPIO43 = MODE0(GPIO43) MODE1(SDA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(11) open-drain +GPIO44 = MODE0(GPIO44) MODE1(SCL2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(11) open-drain +GPIO45 = MODE0(GPIO45) MODE1(SDA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(12) open-drain +GPIO46 = MODE0(GPIO46) MODE1(SCL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(12) open-drain +GPIO47 = MODE0(GPIO47) MODE1(CMDAT2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO48 = MODE0(GPIO48) MODE1(CMDAT3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO49 = MODE0(GPIO49) MODE1(CMDAT4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO50 = MODE0(GPIO50) MODE1(CMDAT5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO51 = MODE0(GPIO51) MODE1(CMDAT6) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO52 = MODE0(GPIO52) MODE1(CMDAT7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO53 = MODE0(GPIO53) MODE1(CMDAT8) MODE2(CMCSD3) MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO54 = MODE0(GPIO54) MODE1(CMDAT9) MODE2(CMCSD2) MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO55 = MODE0(GPIO55) MODE1(CMHSYNC) MODE2(CMCSD1) MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO56 = MODE0(GPIO56) MODE1(CMVSYNC) MODE2(CMCSD0) MODE3() MODE4() MODE5() MODE6() MODE7() - - - +GPIO57 = MODE0(GPIO57) MODE1(MSDC0_DAT0) MODE2(I2S1_WS) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_7_) PU msdc0da_smt - +GPIO58 = MODE0(GPIO58) MODE1(MSDC0_DAT1) MODE2(I2S1_BCK) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_8_) PU msdc0da_smt - +GPIO59 = MODE0(GPIO59) MODE1(MSDC0_DAT2) MODE2(I2S1_MCK) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_9_) PU msdc0da_smt - +GPIO60 = MODE0(GPIO60) MODE1(MSDC0_DAT3) MODE2(I2S1_DO_1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_10_) PU msdc0da_smt - +GPIO61 = MODE0(GPIO61) MODE1(MSDC0_DAT4) MODE2(I2S1_DO_2) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_11_) PU msdc0da_smt - +GPIO62 = MODE0(GPIO62) MODE1(MSDC0_DAT5) MODE2(I2S2_WS) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_12_) PU msdc0da_smt - +GPIO63 = MODE0(GPIO63) MODE1(MSDC0_DAT6) MODE2(I2S2_BCK) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_13_) PU msdc0da_smt - +GPIO64 = MODE0(GPIO64) MODE1(MSDC0_DAT7) MODE2(I2S2_DI_1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_14_) PU msdc0da_smt - +GPIO65 = MODE0(GPIO65) MODE1(MSDC0_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_16_) PD msdc0ck_smt - +GPIO66 = MODE0(GPIO66) MODE1(MSDC0_CMD) MODE2(I2S2_DI_2) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_15_) PU msdc0cm_smt - +GPIO67 = MODE0(GPIO67) MODE1(MSDC0_DSL) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_17_) PD msdc0ds_smt - +GPIO68 = MODE0(GPIO68) MODE1(MSDC0_RSTB) MODE2(I2S2_MCK) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_18_) PU msdc0rs_smt - +GPIO69 = MODE0(GPIO69) MODE1(SPI_CK_0_) MODE2(I2S3_DO_1) MODE3(PWM0) MODE4(PWM5) MODE5(I2S2_MCK) MODE6() MODE7(DBG_MON_B_19_) PD SMT_GROUP(30) - +GPIO70 = MODE0(GPIO70) MODE1(SPI_MI_0_) MODE2(I2S3_DO_2) MODE3(PWM1) MODE4(SPI_MO_0_) MODE5(I2S2_DI_1) MODE6(DSI1_TE) MODE7(DBG_MON_B_20_) PD SMT_GROUP(30) - +GPIO71 = MODE0(GPIO71) MODE1(SPI_MO_0_) MODE2(I2S3_DO_3) MODE3(PWM2) MODE4(SPI_MI_0_) MODE5(I2S2_DI_2) MODE6() MODE7(DBG_MON_B_21_) PD SMT_GROUP(30) - +GPIO72 = MODE0(GPIO72) MODE1(SPI_CS_0_) MODE2(I2S3_DO_4) MODE3(PWM3) MODE4(PWM6) MODE5(DISP_PWM1) MODE6() MODE7(DBG_MON_B_22_) PD SMT_GROUP(30) - +GPIO73 = MODE0(GPIO73) MODE1(MSDC1_DAT0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_24_) PU msdc1da_smt - +GPIO74 = MODE0(GPIO74) MODE1(MSDC1_DAT1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_25_) PU msdc1da_smt - +GPIO75 = MODE0(GPIO75) MODE1(MSDC1_DAT2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_26_) PU msdc1da_smt - +GPIO76 = MODE0(GPIO76) MODE1(MSDC1_DAT3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_27_) PU msdc1da_smt - +GPIO77 = MODE0(GPIO77) MODE1(MSDC1_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_28_) PD msdc1ck_smt - +GPIO78 = MODE0(GPIO78) MODE1(MSDC1_CMD) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_23_) PU msdc1cm_smt - +GPIO79 = MODE0(GPIO79) MODE1(PWRAP_SPIMI) MODE2(PWRAP_SPIMO) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(31) - +GPIO80 = MODE0(GPIO80) MODE1(PWRAP_SPIMO) MODE2(PWRAP_SPIMI) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(31) - +GPIO81 = MODE0(GPIO81) MODE1(PWRAP_SPICK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(31) - +GPIO82 = MODE0(GPIO82) MODE1(PWRAP_SPICS) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(31) - +GPIO83 = MODE0(GPIO83) MODE1(AUD_CLK_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(32) - +GPIO84 = MODE0(GPIO84) MODE1(AUD_DAT_MISO) MODE2(AUD_DAT_MOSI) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(33) - +GPIO85 = MODE0(GPIO85) MODE1(AUD_DAT_MOSI) MODE2(AUD_DAT_MISO) MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(33) - +GPIO86 = MODE0(GPIO86) MODE1(RTC32K_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(34) - +GPIO87 = MODE0(GPIO87) MODE1(DISP_PWM0) MODE2(DISP_PWM1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B_31_) PD SMT_GROUP(34) - +GPIO88 = MODE0(GPIO88) MODE1(SRCLKENAI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(34) - +GPIO89 = MODE0(GPIO89) MODE1(SRCLKENAI2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(34) - +GPIO90 = MODE0(GPIO90) MODE1(SRCLKENA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(34) - +GPIO91 = MODE0(GPIO91) MODE1(SRCLKENA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(34) - +GPIO92 = MODE0(GPIO92) MODE1(PCM1_CLK) MODE2(I2S0_BCK) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_24_) PD SMT_GROUP(13) - +GPIO93 = MODE0(GPIO93) MODE1(PCM1_SYNC) MODE2(I2S0_WS) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_25_) PD SMT_GROUP(14) - +GPIO94 = MODE0(GPIO94) MODE1(PCM1_DI) MODE2(I2S0_DI) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_26_) PD SMT_GROUP(14) - +GPIO95 = MODE0(GPIO95) MODE1(PCM1_DO) MODE2(I2S0_DO) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_27_) PD SMT_GROUP(14) - +GPIO96 = MODE0(GPIO96) MODE1(URXD1) MODE2(UTXD1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_28_) PD SMT_GROUP(15) - +GPIO97 = MODE0(GPIO97) MODE1(UTXD1) MODE2(URXD1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_29_) PU SMT_GROUP(15) - +GPIO98 = MODE0(GPIO98) MODE1(URTS1) MODE2(UCTS1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_30_) PD SMT_GROUP(15) - +GPIO99 = MODE0(GPIO99) MODE1(UCTS1) MODE2(URTS1) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_31_) PD SMT_GROUP(15) - +GPIO100 = MODE0(GPIO100) MODE1(MSDC2_DAT0) MODE2() MODE3(USB_DRVVBUS_P0) MODE4(SDA5) MODE5(USB_DRVVBUS_P1) MODE6() MODE7(DBG_MON_B_0_) PD msdc2da_smt - +GPIO101 = MODE0(GPIO101) MODE1(MSDC2_DAT1) MODE2() MODE3(AUD_SPDIF) MODE4(SCL5) MODE5() MODE6() MODE7(DBG_MON_B_1_) PD msdc2da_smt - +GPIO102 = MODE0(GPIO102) MODE1(MSDC2_DAT2) MODE2() MODE3(UTXD0) MODE4() MODE5(PWM0) MODE6(SPI_CK_1_) MODE7(DBG_MON_B_2_) PD msdc2da_smt - +GPIO103 = MODE0(GPIO103) MODE1(MSDC2_DAT3) MODE2() MODE3(URXD0) MODE4() MODE5(PWM1) MODE6(SPI_MI_1_) MODE7(DBG_MON_B_3_) PD msdc2da_smt - +GPIO104 = MODE0(GPIO104) MODE1(MSDC2_CLK) MODE2() MODE3(UTXD3) MODE4(SDA3) MODE5(PWM2) MODE6(SPI_MO_1_) MODE7(DBG_MON_B_4_) PD msdc2ck_smt - +GPIO105 = MODE0(GPIO105) MODE1(MSDC2_CMD) MODE2() MODE3(URXD3) MODE4(SCL3) MODE5(PWM3) MODE6(SPI_CS_1_) MODE7(DBG_MON_B_5_) PD msdc2cm_smt - +GPIO106 = MODE0(GPIO106) MODE1(SDA3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(20) open-drain +GPIO107 = MODE0(GPIO107) MODE1(SCL3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(20) open-drain +GPIO108 = MODE0(GPIO108) MODE1(JTMS) MODE2(MFG_JTAG_TMS) MODE3() MODE4() MODE5(AP_MD32_JTAG_TMS) MODE6(DFD_TMS) MODE7() PU SMT_GROUP(17) - +GPIO109 = MODE0(GPIO109) MODE1(JTCK) MODE2(MFG_JTAG_TCK) MODE3() MODE4() MODE5(AP_MD32_JTAG_TCK) MODE6(DFD_TCK) MODE7() PU SMT_GROUP(17) - +GPIO110 = MODE0(GPIO110) MODE1(JTDI) MODE2(MFG_JTAG_TDI) MODE3() MODE4() MODE5(AP_MD32_JTAG_TDI) MODE6(DFD_TDI) MODE7() PU SMT_GROUP(17) - +GPIO111 = MODE0(GPIO111) MODE1(JTDO) MODE2(MFG_JTAG_TDO) MODE3() MODE4() MODE5(AP_MD32_JTAG_TDO) MODE6(DFD_TDO) MODE7() PU SMT_GROUP(17) - +GPIO112 = MODE0(GPIO112) MODE1(JTRST_B) MODE2(MFG_JTAG_TRSTN) MODE3() MODE4() MODE5(AP_MD32_JTAG_TRST) MODE6(DFD_NTRST) MODE7() PD SMT_GROUP(17) - +GPIO113 = MODE0(GPIO113) MODE1(URXD0) MODE2(UTXD0) MODE3() MODE4() MODE5() MODE6(I2S2_WS) MODE7(DBG_MON_A_0_) PU SMT_GROUP(18) - +GPIO114 = MODE0(GPIO114) MODE1(UTXD0) MODE2(URXD0) MODE3() MODE4() MODE5() MODE6(I2S2_BCK) MODE7(DBG_MON_A_1_) PU SMT_GROUP(18) - +GPIO115 = MODE0(GPIO115) MODE1(URTS0) MODE2(UCTS0) MODE3() MODE4() MODE5() MODE6(I2S2_MCK) MODE7(DBG_MON_A_2_) PD SMT_GROUP(18) - +GPIO116 = MODE0(GPIO116) MODE1(UCTS0) MODE2(URTS0) MODE3() MODE4() MODE5() MODE6(I2S2_DI_1) MODE7(DBG_MON_A_3_) PD SMT_GROUP(18) - +GPIO117 = MODE0(GPIO117) MODE1(URXD3) MODE2(UTXD3) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_9_) PD SMT_GROUP(21) - +GPIO118 = MODE0(GPIO118) MODE1(UTXD3) MODE2(URXD3) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_10_) PD SMT_GROUP(21) - +GPIO119 = MODE0(GPIO119) MODE1(KROW0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_11_) PD SMT_GROUP(22) - +GPIO120 = MODE0(GPIO120) MODE1(KROW1) MODE2() MODE3(PWM6) MODE4() MODE5() MODE6() MODE7(DBG_MON_A_12_) PD SMT_GROUP(22) - +GPIO121 = MODE0(GPIO121) MODE1(KROW2) MODE2(IRDA_PDN) MODE3(USB_DRVVBUS_P0) MODE4(PWM4) MODE5(USB_DRVVBUS_P1) MODE6() MODE7(DBG_MON_A_13_) PD SMT_GROUP(22) - +GPIO122 = MODE0(GPIO122) MODE1(KCOL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A_14_) PU SMT_GROUP(22) - +GPIO123 = MODE0(GPIO123) MODE1(KCOL1) MODE2(IRDA_RXD) MODE3(PWM5) MODE4() MODE5() MODE6() MODE7(DBG_MON_A_15_) PD SMT_GROUP(22) - +GPIO124 = MODE0(GPIO124) MODE1(KCOL2) MODE2(IRDA_TXD) MODE3(USB_DRVVBUS_P0) MODE4(PWM3) MODE5(USB_DRVVBUS_P1) MODE6() MODE7(DBG_MON_A_16_) PD SMT_GROUP(22) - +GPIO125 = MODE0(GPIO125) MODE1(SDA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(23) - +GPIO126 = MODE0(GPIO126) MODE1(SCL1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(23) - +GPIO127 = MODE0(GPIO127) MODE1(LCM_RST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD SMT_GROUP(16) - +GPIO128 = MODE0(GPIO128) MODE1(I2S0_WS) MODE2(I2S1_WS) MODE3(I2S2_WS) MODE4() MODE5(SPI_CK_2_) MODE6() MODE7(DBG_MON_A_4_) PU SMT_GROUP(40) - +GPIO129 = MODE0(GPIO129) MODE1(I2S0_BCK) MODE2(I2S1_BCK) MODE3(I2S2_BCK) MODE4() MODE5(SPI_MI_2_) MODE6() MODE7(DBG_MON_A_5_) PD SMT_GROUP(41) - +GPIO130 = MODE0(GPIO130) MODE1(I2S0_MCK) MODE2(I2S1_MCK) MODE3(I2S2_MCK) MODE4() MODE5(SPI_MO_2_) MODE6() MODE7(DBG_MON_A_6_) PD SMT_GROUP(41) - +GPIO131 = MODE0(GPIO131) MODE1(I2S0_DO) MODE2(I2S1_DO_1) MODE3(I2S2_DI_1) MODE4() MODE5(SPI_CS_2_) MODE6() MODE7(DBG_MON_A_7_) PD SMT_GROUP(40) - +GPIO132 = MODE0(GPIO132) MODE1(I2S0_DI) MODE2(I2S1_DO_2) MODE3(I2S2_DI_2) MODE4() MODE5() MODE6() MODE7(DBG_MON_A_8_) PD SMT_GROUP(40) - +GPIO133 = MODE0(GPIO133) MODE1(SDA4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(24) - +GPIO134 = MODE0(GPIO134) MODE1(SCL4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU SMT_GROUP(24) - + +[GPIOEXT] +GPIOEXT0 = MODE0(GPIO0) MODE1(INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIOEXT1 = MODE0(GPIO1) MODE1(SRCVOLTEN) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_CK1) MODE7() PD +GPIOEXT2 = MODE0(GPIO2) MODE1(SRCLKEN_PERI) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_CK2) MODE7() PD +GPIOEXT3 = MODE0(GPIO3) MODE1(RTC_32K1V8) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_CK3) MODE7() PU/PD +GPIOEXT4 = MODE0(GPIO4) MODE1(WRAP_EVENT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIOEXT5 = MODE0(GPIO5) MODE1(SPI_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIOEXT6 = MODE0(GPIO6) MODE1(SPI_CSN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIOEXT7 = MODE0(GPIO7) MODE1(SPI_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIOEXT8 = MODE0(GPIO8) MODE1(SPI_MISO) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIOEXT9 = MODE0(GPIO9) MODE1(AUD_CLK) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN0) MODE7(TEST_IN0) PD +GPIOEXT10 = MODE0(GPIO10) MODE1(AUD_MISO) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN1) MODE7(TEST_OUT1) PD +GPIOEXT11 = MODE0(GPIO11) MODE1(AUD_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN2) MODE7(TSET_OUT2) PD +GPIOEXT12 = MODE0(GPIO12) MODE1(COL0_USBDL) MODE2(EINT10_1X) MODE3(PWM1_3X) MODE4() MODE5() MODE6(TEST_IN3) MODE7(TEST_OUT3) PU +GPIOEXT13 = MODE0(GPIO13) MODE1(COL1) MODE2(EINT11_1X) MODE3(SCL0_2X) MODE4() MODE5() MODE6(TEST_IN4) MODE7(TEST_OUT4) PD +GPIOEXT14 = MODE0(GPIO14) MODE1(COL2) MODE2(EINT12_1X) MODE3(SDA0_2X) MODE4() MODE5() MODE6(TEST_IN5) MODE7(TEST_OUT5) PD +GPIOEXT15 = MODE0(GPIO15) MODE1(COL3) MODE2(EINT13_1X) MODE3(SCL1_2X) MODE4() MODE5() MODE6(TEST_IN6) MODE7(TEST_OUT6) PD +GPIOEXT16 = MODE0(GPIO16) MODE1(COL4) MODE2(EINT14_1X) MODE3(SDA1_2X) MODE4() MODE5() MODE6(TEST_IN7) MODE7(TEST_OUT7) PD +GPIOEXT17 = MODE0(GPIO17) MODE1(COL5) MODE2(EINT15_1X) MODE3(SCL2_2X) MODE4() MODE5() MODE6(TEST_IN8) MODE7(TEST_OUT8) PD +GPIOEXT18 = MODE0(GPIO18) MODE1(COL6) MODE2(EINT16_1X) MODE3(SDA2_2X) MODE4(GPIO32K_0) MODE5(GPIO26M_0) MODE6(TEST_IN9) MODE7(TEST_OUT9) PD +GPIOEXT19 = MODE0(GPIO19) MODE1(COL7) MODE2(EINT17_1X) MODE3(PWM2_3X) MODE4(GPIO32K_1) MODE5(GPIO26M_1) MODE6(TEST_IN10) MODE7(TEST_OUT10) PD +GPIOEXT20 = MODE0(GPIO20) MODE1(ROW0) MODE2(EINT18_1X) MODE3(SCL0_3X) MODE4() MODE5() MODE6(TEST_IN11) MODE7(TEST_OUT11) PD +GPIOEXT21 = MODE0(GPIO21) MODE1(ROW1) MODE2(EINT19_1X) MODE3(SDA0_3X) MODE4(AUD_TSTCK) MODE5() MODE6(TEST_IN12) MODE7(TEST_OUT12) PD +GPIOEXT22 = MODE0(GPIO22) MODE1(ROW2) MODE2(EINT20_1X) MODE3(SCL1_3X) MODE4() MODE5() MODE6(TEST_IN13) MODE7(TEST_OUT13) PD +GPIOEXT23 = MODE0(GPIO23) MODE1(ROW3) MODE2(EINT21_1X) MODE3(SDA1_3X) MODE4() MODE5() MODE6(TEST_IN14) MODE7(TEST_OUT14) PD +GPIOEXT24 = MODE0(GPIO24) MODE1(ROW4) MODE2(EINT22_1X) MODE3(SCL2_3X) MODE4() MODE5() MODE6(TEST_IN15) MODE7(TEST_OUT15) PD +GPIOEXT25 = MODE0(GPIO25) MODE1(ROW5) MODE2(EINT23_1X) MODE3(SDA2_3X) MODE4() MODE5() MODE6(TEST_IN16) MODE7(TEST_OUT16) PD +GPIOEXT26 = MODE0(GPIO26) MODE1(ROW6) MODE2(EINT24_1X) MODE3(PWM3_3X) MODE4(GPIO32K_2) MODE5(GPIO26M_2) MODE6(TEST_IN17) MODE7(TEST_OUT17) PD +GPIOEXT27 = MODE0(GPIO27) MODE1(ROW7) MODE2(EINT3_1X) MODE3(CBUS) MODE4(GPIO32K_3) MODE5(GPIO26M_3) MODE6(TEST_IN18) MODE7(TEST_OUT18) PD +GPIOEXT28 = MODE0(GPIO28) MODE1(PWM1) MODE2(EINT4_1X) MODE3() MODE4(GPIO32K_4) MODE5(GPIO26M_4) MODE6(TEST_IN19) MODE7(TEST_OUT19) PD +GPIOEXT29 = MODE0(GPIO29) MODE1(PWM2) MODE2(EINT5_1X) MODE3() MODE4(GPIO32K_5) MODE5(GPIO26M_5) MODE6(TEST_IN20) MODE7(TEST_OUT20) PD +GPIOEXT30 = MODE0(GPIO30) MODE1(PWM3) MODE2(EINT6_1X) MODE3(COL0) MODE4(GPIO32K_6) MODE5(GPIO26M_6) MODE6(TEST_IN21) MODE7(TEST_OUT21) PD +GPIOEXT31 = MODE0(GPIO31) MODE1(SCL0) MODE2(EINT7_1X) MODE3(PWM1_2X) MODE4() MODE5() MODE6(TEST_IN22) MODE7(TEST_OUT22) PU +GPIOEXT32 = MODE0(GPIO32) MODE1(SDA0) MODE2(EINT8_1X) MODE3() MODE4() MODE5() MODE6(TEST_IN23) MODE7(TEST_OUT23) PU +GPIOEXT33 = MODE0(GPIO33) MODE1(SCL1) MODE2(EINT9_1X) MODE3(PWM2_2X) MODE4() MODE5() MODE6(TEST_IN24) MODE7(TEST_OUT24) PU +GPIOEXT34 = MODE0(GPIO34) MODE1(SDA1) MODE2(EINT0_1X) MODE3() MODE4() MODE5() MODE6(TEST_IN25) MODE7(TEST_OUT25) PU +GPIOEXT35 = MODE0(GPIO35) MODE1(SCL2) MODE2(EINT1_1X) MODE3(PWM3_2X) MODE4() MODE5() MODE6(TEST_IN26) MODE7(TEST_OUT26) PU +GPIOEXT36 = MODE0(GPIO36) MODE1(SDA2) MODE2(EINT2_1X) MODE3() MODE4() MODE5() MODE6(TEST_IN27) MODE7(TEST_OUT27) PU +GPIOEXT37 = MODE0(GPIO37) MODE1(HDMISD) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN28) MODE7(TEST_OUT28) PU/PD +GPIOEXT38 = MODE0(GPIO38) MODE1(HDMISCK) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN29) MODE7(TEST_OUT29) PU/PD +GPIOEXT39 = MODE0(GPIO39) MODE1(HTPLG) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN30) MODE7(TEST_OUT30) PU/PD +GPIOEXT40 = MODE0(GPIO40) MODE1(CEC) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN31) MODE7(TEST_OUT31) PU/PD + +[GPO] + +[EINT] +EINT_COUNT = 133 +EINT_DEBOUNCE_TIME_COUNT = 12 + +[EINT_EXT] +EINT_EXT_COUNT = 25 +EINT_EXT_DEBOUNCE_TIME_COUNT = 12 + +[ADC] +ADC_COUNT = 6 + +[ADC_EX_PIN] +0 +1 +12 +13 +14 +15 + +[ADC_EX_PIN] + +[KEYPAD] +KEY_ROW = 8 +KEY_COLUMN = 9 + +[I2C] +I2C_COUNT = 32 +CHANNEL_COUNT = 6 + +[CLK_BUF] +CLK_BUF_COUNT = 4 + +[POWER] +AVDD10_SSUSB +AVDD18_AP +AVDD18_ARMPLL +AVDD18_HDMITX +AVDD18_MEMPLL_A +AVDD18_MEMPLL_B +AVDD18_MIPIRX0 +AVDD18_MIPIRX1 +AVDD18_MIPITX1 +AVDD18_MIPITX2 +AVDD18_PLLGP +AVDD18_SSUSB +AVDD18_USB_P0 +AVDD18_USB_P1 +AVDD33_USB_P0 +AVDD33_USB_P1 +DDRV_A_1 +DDRV_A_2 +DDRV_A_3 +DDRV_A_4 +DDRV_B_1 +DDRV_B_2 +DDRV_B_3 +DDRV_B_4 +DVDD_CORE +DVDD_CORE_1 +DVDD_CORE_2 +DVDD_CORE_3 +DVDD_CORE_4 +DVDD_CORE_5 +DVDD_CORE_6 +DVDD_CORE_7 +DVDD_CORE_8 +DVDD_CORE_9 +DVDD_DVFS1 +DVDD_DVFS2 +DVDD_GPU +DVDD_SRAM1 +DVDD_SRAM2 +DVDD18_EFUSE +VCC12IO_HSIC +VCC18IO_0 +VCC18IO_0_CONN +VCC18IO_0_MSDC3 +VCC18IO_2 +VCC18IO_4 +VCC18IO_5 +VCC18IO_6 +VCC18IO_8 +VCC18IO_EINT +VCC18IO_MC1 +VCC18IO_MC2 +VCC33IO_MC1 +VCC33IO_MC2 +VCC33IO_NAND + +[AVDD10_SSUSB] +VCC1V0 + +[AVDD18_AP] +VIO18 + +[AVDD18_ARMPLL] +VIO18 + +[AVDD18_HDMITX] +VIO18 + +[AVDD18_MEMPLL_A] +VIO18 + +[AVDD18_MEMPLL_B] +VIO18 + +[AVDD18_MIPIRX0] +VIO18 + +[AVDD18_MIPIRX1] +VIO18 + +[AVDD18_MIPITX1] +VIO18 + +[AVDD18_MIPITX2] +VIO18 + +[AVDD18_PLLGP] +VIO18 + +[AVDD18_SSUSB] +VIO18 + +[AVDD18_USB_P0] +VIO18 + +[AVDD18_USB_P1] +VIO18 + +[AVDD33_USB_P0] +VUSB + +[AVDD33_USB_P1] +VUSB + +[DDRV_A_1] +VM + +[DDRV_A_2] +VM + +[DDRV_A_3] +VM + +[DDRV_A_4] +VM + +[DDRV_B_1] +VM + +[DDRV_B_2] +VM + +[DDRV_B_3] +VM + +[DDRV_B_4] +VM + +[DVDD_CORE] +VCORE + +[DVDD_CORE_1] +VCORE + +[DVDD_CORE_2] +VCORE + +[DVDD_CORE_3] +VCORE + +[DVDD_CORE_4] +VCORE + +[DVDD_CORE_5] +VCORE + +[DVDD_CORE_6] +VCORE + +[DVDD_CORE_7] +VCORE + +[DVDD_CORE_8] +VCORE + +[DVDD_CORE_9] +VCORE + +[DVDD_DVFS1] +VDVFS1 + +[DVDD_DVFS2] +VDVFS2 + +[DVDD_GPU] +VGPU + +[DVDD_SRAM1] +VSRAMCA15 + +[DVDD_SRAM2] +VSRAMCA7 + +[DVDD18_EFUSE] +VIO18 + +[VCC12IO_HSIC] +VGP4 + +[VCC18IO_0] +VIO18 + +[VCC18IO_0_CONN] +VIO18 + +[VCC18IO_0_MSDC3] +VIO18 + +[VCC18IO_2] +VIO18 + +[VCC18IO_4] +VIO18 + +[VCC18IO_5] +VIO18 + +[VCC18IO_6] +VIO18 + +[VCC18IO_8] +VIO18 + +[VCC18IO_EINT] +VIO18 + +[VCC18IO_MC1] +VIO18 + +[VCC18IO_MC2] +VIO18 + +[VCC33IO_MC1] +VMC + +[VCC33IO_MC2] +VMC + +[VCC33IO_NAND] +VMC + diff --git a/tools/dct/MT8590.fig b/tools/dct/MT8590.fig new file mode 100755 index 000000000..c8b78a871 --- /dev/null +++ b/tools/dct/MT8590.fig @@ -0,0 +1,445 @@ +[Chip Type] +Chip = MT8590 +GPIO_Pull_Sel = 1 +PMIC_Config = 1 +PMIC_GPIO_Config = 0 +EINT_EXT_Config = 0 +POWER_Config = 1 +POWER_COUNT = 32 +GPIO_ModeNum = 8 +AndroidPhone = 1 +SpecialKey_Config = 1 + + +[GPIO] +GPIO0 = MODE0(GPIO0) MODE1(PWRAP_SPIDO) MODE2(PWRAP_SPIDI) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO1 = MODE0(GPIO1) MODE1(PWRAP_SPIDI) MODE2(PWRAP_SPIDO) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO2 = MODE0(GPIO2) MODE1(PWRAP_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO3 = MODE0(GPIO3) MODE1(PWRAP_SPICK_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO4 = MODE0(GPIO4) MODE1(PWRAP_SPICS_B_I) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO5 = MODE0(GPIO5) MODE1(PWRAP_SPICK2_I) MODE2() MODE3() MODE4() MODE5(ANT_SEL1) MODE6() MODE7() PD +GPIO6 = MODE0(GPIO6) MODE1(PWRAP_SPICS2_B_I) MODE2() MODE3() MODE4() MODE5(ANT_SEL0) MODE6() MODE7(DBG_MON_A[0]) PU +GPIO7 = MODE0(GPIO7) MODE1(SPI1_CS) MODE2() MODE3() MODE4(KCOL0) MODE5() MODE6() MODE7(DBG_MON_B[12]) PD +GPIO8 = MODE0(GPIO8) MODE1(SPI1_MI) MODE2(SPI1_MO) MODE3() MODE4(KCOL1) MODE5() MODE6() MODE7(DBG_MON_B[13]) PD +GPIO9 = MODE0(GPIO9) MODE1(SPI1_MO) MODE2(SPI1_MI) MODE3(EXT_FRAME_SYNC) MODE4(KCOL2) MODE5() MODE6() MODE7(DBG_MON_B[14]) PD +GPIO10 = MODE0(GPIO10) MODE1(RTC32K_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO11 = MODE0(GPIO11) MODE1(WATCHDOG) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO12 = MODE0(GPIO12) MODE1(SRCLKENA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO13 = MODE0(GPIO13) MODE1(SRCLKENAI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO14 = MODE0(GPIO14) MODE1(URXD2) MODE2(UTXD2) MODE3() MODE4() MODE5(SRCCLKENAI2) MODE6() MODE7(DBG_MON_B[30]) PD +GPIO15 = MODE0(GPIO15) MODE1(UTXD2) MODE2(URXD2) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B[31]) PD +GPIO16 = MODE0(GPIO16) MODE1(I2S5_DATA_IN) MODE2() MODE3(PCM_RX) MODE4(ANT_SEL4) MODE5() MODE6() MODE7() PD +GPIO17 = MODE0(GPIO17) MODE1(I2S5_BCK) MODE2() MODE3(PCM_CLK0) MODE4(ANT_SEL2) MODE5() MODE6() MODE7() PD +GPIO18 = MODE0(GPIO18) MODE1(PCM_CLK0) MODE2(MRG_CLK) MODE3() MODE4(MM_TEST_CK) MODE5(CONN_DSP_JCK) MODE6(WCN_PCM_CLKO) MODE7(DBG_MON_A[3]) PD +GPIO19 = MODE0(GPIO19) MODE1(PCM_SYNC) MODE2(MRG_SYNC) MODE3() MODE4() MODE5(CONN_DSP_JINTP) MODE6(WCN_PCM_SYNC) MODE7(DBG_MON_A[5]) PD +GPIO20 = MODE0(GPIO20) MODE1(PCM_RX) MODE2(MRG_RX) MODE3(MRG_TX) MODE4(PCM_TX) MODE5(CONN_DSP_JDI) MODE6(WCN_PCM_RX) MODE7(DBG_MON_A[4]) PD +GPIO21 = MODE0(GPIO21) MODE1(PCM_TX) MODE2(MRG_TX) MODE3(MRG_RX) MODE4(PCM_RX) MODE5(CONN_DSP_JMS) MODE6(WCN_PCM_TX) MODE7(DBG_MON_A[2]) PD +GPIO22 = MODE0(GPIO22) MODE1(UCTS0) MODE2(PCIE0_PERST_N) MODE3(KCOL3) MODE4(CONN_DSP_JDO) MODE5(EXT_FRAME_SYNC) MODE6() MODE7(DBG_MON_A[30]) PD +GPIO23 = MODE0(GPIO23) MODE1(URTS0) MODE2(PCIE1_PERST_N) MODE3(KCOL2) MODE4(CONN_MCU_TDO) MODE5(EXT_FRAME_SYNC) MODE6() MODE7(DBG_MON_A[29]) PD +GPIO24 = MODE0(GPIO24) MODE1(UCTS1) MODE2(PCIE2_PERST_N) MODE3(KCOL1) MODE4(CONN_MCU_DBGACK_N) MODE5() MODE6() MODE7(DBG_MON_A[28]) PD +GPIO25 = MODE0(GPIO25) MODE1(URTS1) MODE2() MODE3(KCOL0) MODE4(CONN_MCU_DBGI_N) MODE5() MODE6() MODE7(DBG_MON_A[27]) PD +GPIO26 = MODE0(GPIO26) MODE1(UCTS3) MODE2(DRV_VBUS_P1) MODE3(KROW3) MODE4(CONN_MCU_TCK0) MODE5(CONN_MCU_AICE_JCKC) MODE6(PCIE2_WAKE_N) MODE7(DBG_MON_A[26]) PD +GPIO27 = MODE0(GPIO27) MODE1(URTS3) MODE2(IDDIG_P1) MODE3(KROW2) MODE4(CONN_MCU_TDI) MODE5() MODE6(PCIE1_WAKE_N) MODE7(DBG_MON_A[25]) PD +GPIO28 = MODE0(GPIO28) MODE1(DRV_VBUS) MODE2() MODE3(KROW1) MODE4(CONN_MCU_TRST_B) MODE5() MODE6(PCIE0_WAKE_N) MODE7(DBG_MON_A[24]) PD +GPIO29 = MODE0(GPIO29) MODE1(IDDIG) MODE2(MSDC1_WP) MODE3(KROW0) MODE4(CONN_MCU_TMS) MODE5(CONN_MCU_AICE_JMSC) MODE6(PCIE2_PERST_N) MODE7(DBG_MON_A[23]) PD +GPIO30 = MODE0(GPIO30) MODE1(I2S5_LRCK) MODE2() MODE3(PCM_SYNC) MODE4(ANT_SEL1) MODE5() MODE6() MODE7() PD +GPIO31 = MODE0(GPIO31) MODE1(I2S5_MCLK) MODE2() MODE3() MODE4(ANT_SEL0) MODE5() MODE6() MODE7() PD +GPIO32 = MODE0(GPIO32) MODE1(I2S5_DATA) MODE2(I2S5_DATA_BYPS) MODE3(PCM_TX) MODE4(ANT_SEL3) MODE5() MODE6() MODE7() PD +GPIO33 = MODE0(GPIO33) MODE1(I2S1_DATA) MODE2(I2S1_DATA_BYPS) MODE3(PCM_TX) MODE4(IMG_TEST_CK) MODE5(G1_RXD0) MODE6(WCN_PCM_TX) MODE7(DBG_MON_B[8]) PD +GPIO34 = MODE0(GPIO34) MODE1(I2S1_DATA_IN) MODE2() MODE3(PCM_RX) MODE4(VDEC_TEST_CK) MODE5(G1_RXD1) MODE6(WCN_PCM_RX) MODE7(DBG_MON_B[7]) PD +GPIO35 = MODE0(GPIO35) MODE1(I2S1_BCK) MODE2() MODE3(PCM_CLK0) MODE4() MODE5(G1_RXD2) MODE6(WCN_PCM_CLKO) MODE7(DBG_MON_B[9]) PD +GPIO36 = MODE0(GPIO36) MODE1(I2S1_LRCK) MODE2() MODE3(PCM_SYNC) MODE4() MODE5(G1_RXD3) MODE6(WCN_PCM_SYNC) MODE7(DBG_MON_B[10]) PD +GPIO37 = MODE0(GPIO37) MODE1(I2S1_MCLK) MODE2() MODE3() MODE4() MODE5(G1_RXDV) MODE6() MODE7(DBG_MON_B[11]) PD +GPIO38 = MODE0(GPIO38) MODE1(I2S2_DATA) MODE2(I2S2_DATA_BYPS) MODE3(PCM_TX) MODE4(DMIC_DAT0) MODE5() MODE6() MODE7() PD +GPIO39 = MODE0(GPIO39) MODE1(JTMS) MODE2(CONN_MCU_TMS) MODE3(CONN_MCU_AICE_JMSC) MODE4(DFD_TMS_XI) MODE5() MODE6() MODE7() PU +GPIO40 = MODE0(GPIO40) MODE1(JTCK) MODE2(CONN_MCU_TCK1) MODE3(CONN_MCU_AICE_JCKC) MODE4(DFD_TCK_XI) MODE5() MODE6() MODE7() PU +GPIO41 = MODE0(GPIO41) MODE1(JTDI) MODE2(CONN_MCU_TDI) MODE3() MODE4(DFD_TDI_XI) MODE5() MODE6() MODE7() PU +GPIO42 = MODE0(GPIO42) MODE1(JTDO) MODE2(CONN_MCU_TDO) MODE3() MODE4(DFD_TDO) MODE5() MODE6() MODE7() PU +GPIO43 = MODE0(GPIO43) MODE1(NCLE) MODE2(EXT_XCS2) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO44 = MODE0(GPIO44) MODE1(NCEB1) MODE2(IDDIG) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO45 = MODE0(GPIO45) MODE1(NCEB0) MODE2(DRV_VBUS) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO46 = MODE0(GPIO46) MODE1(IR) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO47 = MODE0(GPIO47) MODE1(NREB) MODE2(IDDIG_P1) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO48 = MODE0(GPIO48) MODE1(NRNB) MODE2(DRV_VBUS_P1) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO49 = MODE0(GPIO49) MODE1(I2S0_DATA) MODE2(I2S0_DATA_BYPS) MODE3(PCM_TX) MODE4() MODE5() MODE6(WCN_I2S_DO) MODE7(DBG_MON_B[3]) PD +GPIO50 = MODE0(GPIO50) MODE1(I2S2_BCK) MODE2() MODE3(PCM_CLK0) MODE4(DMIC_SCK1) MODE5() MODE6() MODE7() PD +GPIO51 = MODE0(GPIO51) MODE1(I2S2_DATA_IN) MODE2() MODE3(PCM_RX) MODE4(DMIC_SCK0) MODE5() MODE6() MODE7() PD +GPIO52 = MODE0(GPIO52) MODE1(I2S2_LRCK) MODE2() MODE3(PCM_SYNC) MODE4(DMIC_DAT1) MODE5() MODE6() MODE7() PD +GPIO53 = MODE0(GPIO53) MODE1(SPI0_CS) MODE2() MODE3(SPDIF) MODE4(ADC_CK) MODE5(PWM1) MODE6() MODE7(DBG_MON_A[7]) PD +GPIO54 = MODE0(GPIO54) MODE1(SPI0_CK) MODE2() MODE3(SPDIF_IN1) MODE4(ADC_DAT_IN) MODE5() MODE6() MODE7(DBG_MON_A[10]) PD +GPIO55 = MODE0(GPIO55) MODE1(SPI0_MI) MODE2(SPI0_MO) MODE3(MSDC1_WP) MODE4(ADC_WS) MODE5(PWM2) MODE6() MODE7(DBG_MON_A[8]) PD +GPIO56 = MODE0(GPIO56) MODE1(SPI0_MO) MODE2(SPI0_MI) MODE3(SPDIF_IN0) MODE4() MODE5() MODE6() MODE7(DBG_MON_A[9]) PD +GPIO57 = MODE0(GPIO57) MODE1(SDA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO58 = MODE0(GPIO58) MODE1(SCL1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO59 = MODE0(GPIO59) MODE1(RAMBUF_I_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO60 = MODE0(GPIO60) MODE1(WB_RSTB) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[11]) PD +GPIO61 = MODE0(GPIO61) MODE1(F2W_DATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[16]) PD +GPIO62 = MODE0(GPIO62) MODE1(F2W_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[15]) PD +GPIO63 = MODE0(GPIO63) MODE1(WB_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[13]) PD +GPIO64 = MODE0(GPIO64) MODE1(WB_SDATA) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[12]) PD +GPIO65 = MODE0(GPIO65) MODE1(WB_SEN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[14]) PD +GPIO66 = MODE0(GPIO66) MODE1(WB_CRTL0) MODE2() MODE3() MODE4() MODE5(DFD_NTRST_XI) MODE6() MODE7(DBG_MON_A[17]) PD +GPIO67 = MODE0(GPIO67) MODE1(WB_CRTL1) MODE2() MODE3() MODE4() MODE5(DFD_TMS_XI) MODE6() MODE7(DBG_MON_A[18]) PD +GPIO68 = MODE0(GPIO68) MODE1(WB_CRTL2) MODE2() MODE3() MODE4() MODE5(DFD_TCK_XI) MODE6() MODE7(DBG_MON_A[19]) PD +GPIO69 = MODE0(GPIO69) MODE1(WB_CRTL3) MODE2() MODE3() MODE4() MODE5(DFD_TDI_XI) MODE6() MODE7(DBG_MON_A[20]) PD +GPIO70 = MODE0(GPIO70) MODE1(WB_CRTL4) MODE2() MODE3() MODE4() MODE5(DFD_TDO) MODE6() MODE7(DBG_MON_A[2]) PD +GPIO71 = MODE0(GPIO71) MODE1(WB_CRTL5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[22]) PD +GPIO72 = MODE0(GPIO72) MODE1(I2S0_DATA_IN) MODE2() MODE3(PCM_RX) MODE4(PWM0) MODE5(DISP_PWM) MODE6(WCN_I2S_DI) MODE7(DBG_MON_B[2]) PD +GPIO73 = MODE0(GPIO73) MODE1(I2S0_LRCK) MODE2() MODE3(PCM_SYNC) MODE4() MODE5() MODE6(WCN_I2S_LRCK) MODE7(DBG_MON_B[5]) PD +GPIO74 = MODE0(GPIO74) MODE1(I2S0_BCK) MODE2() MODE3(PCM_CLK0) MODE4() MODE5() MODE6(WCN_I2S_BCK) MODE7(DBG_MON_B[4]) PD +GPIO75 = MODE0(GPIO75) MODE1(SDA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO76 = MODE0(GPIO76) MODE1(SCL0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO77 = MODE0(GPIO77) MODE1(SDA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO78 = MODE0(GPIO78) MODE1(SCL2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO79 = MODE0(GPIO79) MODE1(URXD0) MODE2(UTXD0) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO80 = MODE0(GPIO80) MODE1(UTXD0) MODE2(URXD0) MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO81 = MODE0(GPIO81) MODE1(URXD1) MODE2(UTXD1) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO82 = MODE0(GPIO82) MODE1(UTXD1) MODE2(URXD1) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO83 = MODE0(GPIO83) MODE1(LCM_RST) MODE2(VDAC_CK_XI) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B[1]) PD +GPIO84 = MODE0(GPIO84) MODE1(DSI_TE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B[0]) PD +GPIO85 = MODE0(GPIO85) MODE1(MSDC2_CMD) MODE2(ANT_SEL0) MODE3(SDA1) MODE4() MODE5() MODE6(I2SOUT_BCK) MODE7() PD +GPIO86 = MODE0(GPIO86) MODE1(MSDC2_CLK) MODE2(ANT_SEL1) MODE3(SCL1) MODE4() MODE5() MODE6(I2SOUT_LRCK) MODE7() PD +GPIO87 = MODE0(GPIO87) MODE1(MSDC2_DAT0) MODE2(ANT_SEL2) MODE3() MODE4() MODE5(UTXD0) MODE6(I2SOUT_DATA_OUT) MODE7() PD +GPIO88 = MODE0(GPIO88) MODE1(MSDC2_DAT1) MODE2(ANT_SEL3) MODE3(PWM0) MODE4() MODE5(URXD0) MODE6(PWM1) MODE7() PD +GPIO89 = MODE0(GPIO89) MODE1(MSDC2_DAT2) MODE2(ANT_SEL4) MODE3(SDA2) MODE4() MODE5(UTXD1) MODE6(PWM2) MODE7() PD +GPIO90 = MODE0(GPIO90) MODE1(MSDC2_DAT3) MODE2(ANT_SEL5) MODE3(SCL2) MODE4(EXT_FRAME_SYNC) MODE5(URXD1) MODE6(PWM3) MODE7() PD +GPIO91 = MODE0(GPIO91) MODE1(TDN3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO92 = MODE0(GPIO92) MODE1(TDP3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO93 = MODE0(GPIO93) MODE1(TDN2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO94 = MODE0(GPIO94) MODE1(TDP2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO95 = MODE0(GPIO95) MODE1(TCN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO96 = MODE0(GPIO96) MODE1(TCP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO97 = MODE0(GPIO97) MODE1(TDN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO98 = MODE0(GPIO98) MODE1(TDP1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO99 = MODE0(GPIO99) MODE1(TDN0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO100 = MODE0(GPIO100) MODE1(TDP0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO101 = MODE0(GPIO101) MODE1(SPI2_CS) MODE2() MODE3(SCL3) MODE4(KROW0) MODE5() MODE6() MODE7() PD +GPIO102 = MODE0(GPIO102) MODE1(SPI2_MI) MODE2(SPI2_MO) MODE3(SDA3) MODE4(KROW1) MODE5() MODE6() MODE7() PD +GPIO103 = MODE0(GPIO103) MODE1(SPI2_MO) MODE2(SPI2_MI) MODE3(SCL3) MODE4(KROW2) MODE5() MODE6() MODE7() PD +GPIO104 = MODE0(GPIO104) MODE1(SPI2_CK) MODE2() MODE3(SDA3) MODE4(KROW3) MODE5() MODE6() MODE7() PD +GPIO105 = MODE0(GPIO105) MODE1(MSDC1_CMD) MODE2(ANT_SEL0) MODE3(SDA1) MODE4() MODE5() MODE6(I2SOUT_BCK) MODE7(DBG_MON_B[27]) PU +GPIO106 = MODE0(GPIO106) MODE1(MSDC1_CLK) MODE2(ANT_SEL1) MODE3(SCL1) MODE4() MODE5() MODE6(I2SOUT_LRCK) MODE7(DBG_MON_B[28]) PD +GPIO107 = MODE0(GPIO107) MODE1(MSDC1_DAT0) MODE2(ANT_SEL2) MODE3() MODE4() MODE5(UTXD0) MODE6(I2SOUT_DATA_OUT) MODE7(DBG_MON_B[26]) PU +GPIO108 = MODE0(GPIO108) MODE1(MSDC1_DAT1) MODE2(ANT_SEL3) MODE3(PWM0) MODE4() MODE5(URXD0) MODE6(PWM1) MODE7(DBG_MON_B[25]) PU +GPIO109 = MODE0(GPIO109) MODE1(MSDC1_DAT2) MODE2(ANT_SEL4) MODE3(SDA2) MODE4() MODE5(UTXD1) MODE6(PWM2) MODE7(DBG_MON_B[24]) PU +GPIO110 = MODE0(GPIO110) MODE1(MSDC1_DAT3) MODE2(ANT_SEL5) MODE3(SCL2) MODE4(EXT_FRAME_SYNC) MODE5(URXD1) MODE6(PWM3) MODE7(DBG_MON_B[23]) PU +GPIO111 = MODE0(GPIO111) MODE1(MSDC0_DAT7) MODE2() MODE3() MODE4(NLD7) MODE5() MODE6() MODE7() PU +GPIO112 = MODE0(GPIO112) MODE1(MSDC0_DAT6) MODE2() MODE3() MODE4(NLD6) MODE5() MODE6() MODE7() PU +GPIO113 = MODE0(GPIO113) MODE1(MSDC0_DAT5) MODE2() MODE3() MODE4(NLD5) MODE5() MODE6() MODE7() PU +GPIO114 = MODE0(GPIO114) MODE1(MSDC0_DAT4) MODE2() MODE3() MODE4(NLD4) MODE5() MODE6() MODE7() PU +GPIO115 = MODE0(GPIO115) MODE1(MSDC0_RSTB) MODE2() MODE3() MODE4(NLD8) MODE5() MODE6() MODE7() PU +GPIO116 = MODE0(GPIO116) MODE1(MSDC0_CMD) MODE2() MODE3() MODE4(NALE) MODE5() MODE6() MODE7() PU +GPIO117 = MODE0(GPIO117) MODE1(MSDC0_CLK) MODE2() MODE3() MODE4(NWEB) MODE5() MODE6() MODE7() PD +GPIO118 = MODE0(GPIO118) MODE1(MSDC0_DAT3) MODE2() MODE3() MODE4(NLD3) MODE5() MODE6() MODE7() PU +GPIO119 = MODE0(GPIO119) MODE1(MSDC0_DAT2) MODE2() MODE3() MODE4(NLD2) MODE5() MODE6() MODE7() PU +GPIO120 = MODE0(GPIO120) MODE1(MSDC0_DAT1) MODE2() MODE3() MODE4(NLD1) MODE5() MODE6() MODE7() PU +GPIO121 = MODE0(GPIO121) MODE1(MSDC0_DAT0) MODE2() MODE3() MODE4(NLD0) MODE5(WATCHDOG) MODE6() MODE7() PU +GPIO122 = MODE0(GPIO122) MODE1(CEC) MODE2() MODE3() MODE4(SDA2) MODE5(URXD0) MODE6() MODE7() PU +GPIO123 = MODE0(GPIO123) MODE1(HTPLG) MODE2() MODE3() MODE4(SCL2) MODE5(UTXD0) MODE6() MODE7() PU +GPIO124 = MODE0(GPIO124) MODE1(HDMISCK) MODE2() MODE3() MODE4(SDA1) MODE5(PWM3) MODE6() MODE7() PU +GPIO125 = MODE0(GPIO125) MODE1(HDMISD) MODE2() MODE3() MODE4(SCL1) MODE5(PWM4) MODE6() MODE7() PU +GPIO126 = MODE0(GPIO126) MODE1(I2S0_MCLK) MODE2() MODE3() MODE4() MODE5() MODE6(WCN_I2S_MCLK) MODE7(DBG_MON_B[6]) PD +GPIO127 = MODE0(GPIO127) MODE1(RAMBUF_IDATA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO128 = MODE0(GPIO128) MODE1(RAMBUF_IDATA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO129 = MODE0(GPIO129) MODE1(RAMBUF_IDATA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO130 = MODE0(GPIO130) MODE1(RAMBUF_IDATA3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO131 = MODE0(GPIO131) MODE1(RAMBUF_IDATA4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO132 = MODE0(GPIO132) MODE1(RAMBUF_IDATA5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO133 = MODE0(GPIO133) MODE1(RAMBUF_IDATA6) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO134 = MODE0(GPIO134) MODE1(RAMBUF_IDATA7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO135 = MODE0(GPIO135) MODE1(RAMBUF_IDATA8) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO136 = MODE0(GPIO136) MODE1(RAMBUF_IDATA9) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO137 = MODE0(GPIO137) MODE1(RAMBUF_IDATA10) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO138 = MODE0(GPIO138) MODE1(RAMBUF_IDATA11) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO139 = MODE0(GPIO139) MODE1(RAMBUF_IDATA12) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO140 = MODE0(GPIO140) MODE1(RAMBUF_IDATA13) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO141 = MODE0(GPIO141) MODE1(RAMBUF_IDATA14) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO142 = MODE0(GPIO142) MODE1(RAMBUF_IDATA15) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO143 = MODE0(GPIO143) MODE1(RAMBUF_ODATA0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO144 = MODE0(GPIO144) MODE1(RAMBUF_ODATA1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO145 = MODE0(GPIO145) MODE1(RAMBUF_ODATA2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO146 = MODE0(GPIO146) MODE1(RAMBUF_ODATA3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO147 = MODE0(GPIO147) MODE1(RAMBUF_ODATA4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO148 = MODE0(GPIO148) MODE1(RAMBUF_ODATA5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO149 = MODE0(GPIO149) MODE1(RAMBUF_ODATA6) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO150 = MODE0(GPIO150) MODE1(RAMBUF_ODATA7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO151 = MODE0(GPIO151) MODE1(RAMBUF_ODATA8) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO152 = MODE0(GPIO152) MODE1(RAMBUF_ODATA9) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO153 = MODE0(GPIO153) MODE1(RAMBUF_ODATA10) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO154 = MODE0(GPIO154) MODE1(RAMBUF_ODATA11) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO155 = MODE0(GPIO155) MODE1(RAMBUF_ODATA12) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO156 = MODE0(GPIO156) MODE1(RAMBUF_ODATA13) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO157 = MODE0(GPIO157) MODE1(RAMBUF_ODATA14) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO158 = MODE0(GPIO158) MODE1(RAMBUF_ODATA15) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO159 = MODE0(GPIO159) MODE1(RAMBUF_BE0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO160 = MODE0(GPIO160) MODE1(RAMBUF_BE1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO161 = MODE0(GPIO161) MODE1(AP2PT_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO162 = MODE0(GPIO162) MODE1(AP2PT_INT_CLR) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO163 = MODE0(GPIO163) MODE1(PT2AP_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO164 = MODE0(GPIO164) MODE1(PT2AP_INT_CLR) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO165 = MODE0(GPIO165) MODE1(AP2UP_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO166 = MODE0(GPIO166) MODE1(AP2UP_INT_CLR) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO167 = MODE0(GPIO167) MODE1(UP2AP_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO168 = MODE0(GPIO168) MODE1(UP2AP_INT_CLR) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO169 = MODE0(GPIO169) MODE1(RAMBUF_ADDR0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO170 = MODE0(GPIO170) MODE1(RAMBUF_ADDR1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO171 = MODE0(GPIO171) MODE1(RAMBUF_ADDR2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO172 = MODE0(GPIO172) MODE1(RAMBUF_ADDR3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO173 = MODE0(GPIO173) MODE1(RAMBUF_ADDR4) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO174 = MODE0(GPIO174) MODE1(RAMBUF_ADDR5) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO175 = MODE0(GPIO175) MODE1(RAMBUF_ADDR6) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO176 = MODE0(GPIO176) MODE1(RAMBUF_ADDR7) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO177 = MODE0(GPIO177) MODE1(RAMBUF_ADDR8) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO178 = MODE0(GPIO178) MODE1(RAMBUF_ADDR9) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO179 = MODE0(GPIO179) MODE1(RAMBUF_ADDR10) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO180 = MODE0(GPIO180) MODE1(RAMBUF_RW) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO181 = MODE0(GPIO181) MODE1(RAMBUF_LAST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO182 = MODE0(GPIO182) MODE1(RAMBUF_HP) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO183 = MODE0(GPIO183) MODE1(RAMBUF_REQ) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO184 = MODE0(GPIO184) MODE1(RAMBUF_ALE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO185 = MODE0(GPIO185) MODE1(RAMBUF_DLE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO186 = MODE0(GPIO186) MODE1(RAMBUF_WDLE) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO187 = MODE0(GPIO187) MODE1(RAMBUF_O_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO188 = MODE0(GPIO188) MODE1(I2S2_MCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO189 = MODE0(GPIO189) MODE1(I2S3_DATA) MODE2(I2S3_DATA_BYPS) MODE3(PCM_TX) MODE4() MODE5() MODE6() MODE7() PD +GPIO190 = MODE0(GPIO190) MODE1(I2S3_DATA_IN) MODE2() MODE3(PCM_RX) MODE4() MODE5() MODE6() MODE7() PD +GPIO191 = MODE0(GPIO191) MODE1(I2S3_BCK) MODE2() MODE3(PCM_CLK0) MODE4() MODE5() MODE6() MODE7() PD +GPIO192 = MODE0(GPIO192) MODE1(I2S3_LRCK) MODE2() MODE3(PCM_SYNC) MODE4() MODE5() MODE6() MODE7() PD +GPIO193 = MODE0(GPIO193) MODE1(I2S3_MCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO194 = MODE0(GPIO194) MODE1(I2S4_DATA) MODE2(I2S4_DATA_BYPS) MODE3(PCM_TX) MODE4() MODE5() MODE6() MODE7() PD +GPIO195 = MODE0(GPIO195) MODE1(I2S4_DATA_IN) MODE2() MODE3(PCM_RX) MODE4() MODE5() MODE6() MODE7() PD +GPIO196 = MODE0(GPIO196) MODE1(I2S4_BCK) MODE2() MODE3(PCM_CLK0) MODE4() MODE5() MODE6() MODE7() PD +GPIO197 = MODE0(GPIO197) MODE1(I2S4_LRCK) MODE2() MODE3(PCM_SYNC) MODE4() MODE5() MODE6() MODE7() PD +GPIO198 = MODE0(GPIO198) MODE1(I2S4_MCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO199 = MODE0(GPIO199) MODE1(SPI1_CK) MODE2() MODE3(EXT_FRAME_SYNC) MODE4(KCOL3) MODE5() MODE6() MODE7(DBG_MON_B[15]) PD +GPIO200 = MODE0(GPIO200) MODE1(SPDIF_OUT) MODE2() MODE3() MODE4() MODE5(G1_TXD3) MODE6(URXD2) MODE7(DBG_MON_B[16]) PD +GPIO201 = MODE0(GPIO201) MODE1(SPDIF_IN0) MODE2() MODE3() MODE4() MODE5(G1_TXEN) MODE6(UTXD2) MODE7(DBG_MON_B[17]) PD +GPIO202 = MODE0(GPIO202) MODE1(SPDIF_IN1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO203 = MODE0(GPIO203) MODE1(PWM0) MODE2(DISP_PWM) MODE3() MODE4() MODE5(G1_TXD2) MODE6() MODE7(DBG_MON_B[18]) PD +GPIO204 = MODE0(GPIO204) MODE1(PWM1) MODE2(CLKM3) MODE3() MODE4() MODE5(G1_TXD1) MODE6() MODE7(DBG_MON_B[19]) PD +GPIO205 = MODE0(GPIO205) MODE1(PWM2) MODE2(CLKM2) MODE3() MODE4() MODE5(G1_TXD0) MODE6() MODE7(DBG_MON_B[20]) PD +GPIO206 = MODE0(GPIO206) MODE1(PWM3) MODE2(CLKM1) MODE3(EXT_FRAME_SYNC) MODE4() MODE5(G1_TXC) MODE6() MODE7(DBG_MON_B[21]) PD +GPIO207 = MODE0(GPIO207) MODE1(PWM4) MODE2(CLKM0) MODE3(EXT_FRAME_SYNC) MODE4() MODE5(G1_RXC) MODE6() MODE7(DBG_MON_B[22]) PD +GPIO208 = MODE0(GPIO208) MODE1(AUD_EXT_CK1) MODE2(PWM0) MODE3(PCIE0_PERST_N) MODE4(ANT_SEL5) MODE5(DISP_PWM) MODE6() MODE7(DBG_MON_A[31]) PD +GPIO209 = MODE0(GPIO209) MODE1(AUD_EXT_CK2) MODE2(MSDC1_WP) MODE3(PCIE1_PERST_N) MODE4() MODE5(PWM1) MODE6() MODE7(DBG_MON_A[32]) PD +GPIO210 = MODE0(GPIO210) MODE1(AUD_CLOCK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO211 = MODE0(GPIO211) MODE1(DVP_RESET) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO212 = MODE0(GPIO212) MODE1(DVP_CLOCK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO213 = MODE0(GPIO213) MODE1(DVP_CS) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO214 = MODE0(GPIO214) MODE1(DVP_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO215 = MODE0(GPIO215) MODE1(DVP_DI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO216 = MODE0(GPIO216) MODE1(DVP_DO) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO217 = MODE0(GPIO217) MODE1(AP_CS) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO218 = MODE0(GPIO218) MODE1(AP_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO219 = MODE0(GPIO219) MODE1(AP_DI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO220 = MODE0(GPIO220) MODE1(AP_DO) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO221 = MODE0(GPIO221) MODE1(DVD_BCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO222 = MODE0(GPIO222) MODE1(T8032_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO223 = MODE0(GPIO223) MODE1(AP_BCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO224 = MODE0(GPIO224) MODE1(HOST_CS) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO225 = MODE0(GPIO225) MODE1(HOST_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO226 = MODE0(GPIO226) MODE1(HOST_DO0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO227 = MODE0(GPIO227) MODE1(HOST_DO1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO228 = MODE0(GPIO228) MODE1(SLV_CS) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO229 = MODE0(GPIO229) MODE1(SLV_CK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO230 = MODE0(GPIO230) MODE1(SLV_DI0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO231 = MODE0(GPIO231) MODE1(SLV_DI1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO232 = MODE0(GPIO232) MODE1(AP2DSP_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO233 = MODE0(GPIO233) MODE1(AP2DSP_INT_CLR) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO234 = MODE0(GPIO234) MODE1(DSP2AP_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO235 = MODE0(GPIO235) MODE1(DSP2AP_INT_CLR) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO236 = MODE0(GPIO236) MODE1(EXT_SDIO3) MODE2(IDDIG) MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_A[1]) PD +GPIO237 = MODE0(GPIO237) MODE1(EXT_SDIO2) MODE2(DRV_VBUS) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO238 = MODE0(GPIO238) MODE1(EXT_SDIO1) MODE2(IDDIG_P1) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO239 = MODE0(GPIO239) MODE1(EXT_SDIO0) MODE2(DRV_VBUS_P1) MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO240 = MODE0(GPIO240) MODE1(EXT_XCS) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO241 = MODE0(GPIO241) MODE1(EXT_SCK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO242 = MODE0(GPIO242) MODE1(URTS2) MODE2(UTXD3) MODE3(URXD3) MODE4(SCL1) MODE5() MODE6() MODE7(DBG_MON_B[32]) PD +GPIO243 = MODE0(GPIO243) MODE1(UCTS2) MODE2(URXD3) MODE3(UTXD3) MODE4(SDA1) MODE5() MODE6() MODE7(DBG_MON_A[6]) PD +GPIO244 = MODE0(GPIO244) MODE1(HDMI_SDA_RX) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO245 = MODE0(GPIO245) MODE1(HDMI_SCL_RX) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO246 = MODE0(GPIO246) MODE1() MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO247 = MODE0(GPIO247) MODE1(HDMI_HPD_RX) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO248 = MODE0(GPIO248) MODE1(HDMI_TESTOUTP_RX) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO249 = MODE0(GPIO249) MODE1(MSDC0E_RSTB) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO250 = MODE0(GPIO250) MODE1(MSDC3_DAT7) MODE2() MODE3() MODE4() MODE5() MODE6(PCIE0_CLKREQ_N) MODE7() PU +GPIO251 = MODE0(GPIO251) MODE1(MSDC3_DAT6) MODE2() MODE3() MODE4() MODE5() MODE6(PCIE0_WAKE_N) MODE7() PU +GPIO252 = MODE0(GPIO252) MODE1(MSDC3_DAT5) MODE2() MODE3() MODE4() MODE5() MODE6(PCIE1_CLKREQ_N) MODE7() PU +GPIO253 = MODE0(GPIO253) MODE1(MSDC3_DAT4) MODE2() MODE3() MODE4() MODE5() MODE6(PCIE1_WAKE_N) MODE7() PU +GPIO254 = MODE0(GPIO254) MODE1(MSDC3_DAT3) MODE2() MODE3() MODE4() MODE5() MODE6(PCIE2_CLKREQ_N) MODE7() PU +GPIO255 = MODE0(GPIO255) MODE1(MSDC3_DAT2) MODE2() MODE3() MODE4() MODE5() MODE6(PCIE2_WAKE_N) MODE7() PU +GPIO256 = MODE0(GPIO256) MODE1(MSDC3_DAT1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO257 = MODE0(GPIO257) MODE1(MSDC3_DAT0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO258 = MODE0(GPIO258) MODE1(MSDC3_CMD) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO259 = MODE0(GPIO259) MODE1(MSDC3_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO260 = MODE0(GPIO260) MODE1(MSDC3_DSL) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO261 = MODE0(GPIO261) MODE1(MSDC1_INS) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7(DBG_MON_B[29]) PD +GPIO262 = MODE0(GPIO262) MODE1(G2_TXEN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO263 = MODE0(GPIO263) MODE1(G2_TXD3) MODE2() MODE3() MODE4() MODE5() MODE6(ANT_SEL5) MODE7() PD +GPIO264 = MODE0(GPIO264) MODE1(G2_TXD2) MODE2() MODE3() MODE4() MODE5() MODE6(ANT_SEL4) MODE7() PD +GPIO265 = MODE0(GPIO265) MODE1(G2_TXD1) MODE2() MODE3() MODE4() MODE5() MODE6(ANT_SEL3) MODE7() PD +GPIO266 = MODE0(GPIO266) MODE1(G2_TXD0) MODE2() MODE3() MODE4() MODE5() MODE6(ANT_SEL2) MODE7() PD +GPIO267 = MODE0(GPIO267) MODE1(G2_TXC) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO268 = MODE0(GPIO268) MODE1(G2_RXC) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO269 = MODE0(GPIO269) MODE1(G2_RXD0) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO270 = MODE0(GPIO270) MODE1(G2_RXD1) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO271 = MODE0(GPIO271) MODE1(G2_RXD2) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO272 = MODE0(GPIO272) MODE1(G2_RXD3) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO273 = MODE0(GPIO273) MODE1(ESW_INT) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO274 = MODE0(GPIO274) MODE1(G2_RXDV) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO275 = MODE0(GPIO275) MODE1(MDC) MODE2() MODE3() MODE4() MODE5() MODE6(ANT_SEL0) MODE7() PD +GPIO276 = MODE0(GPIO276) MODE1(MDIO) MODE2() MODE3() MODE4() MODE5() MODE6(ANT_SEL1) MODE7() PD +GPIO277 = MODE0(GPIO277) MODE1(ESW_RST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO278 = MODE0(GPIO278) MODE1(JTAG_RESET) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO279 = MODE0(GPIO279) MODE1(USB3_RES_BOND) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD + + +[GPIOEXT] +GPIO0 = MODE0(GPIO0) MODE1(INT) MODE2() MODE3() MODE4() MODE5(TEST_CK2) MODE6(TEST_IN5) MODE7(TEST_OUT5) PD +GPIO1 = MODE0(GPIO1) MODE1(SRCLKEN) MODE2() MODE3() MODE4() MODE5(TEST_CK0) MODE6(TEST_IN4) MODE7(TEST_OUT4) PD +GPIO2 = MODE0(GPIO2) MODE1(RTC_32K1V8) MODE2() MODE3() MODE4() MODE5(TEST_CK1) MODE6(TEST_IN3) MODE7(TEST_OUT3) -- +GPIO3 = MODE0(GPIO3) MODE1(SPI_CLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO4 = MODE0(GPIO4) MODE1(SPI_CSN) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PU +GPIO5 = MODE0(GPIO5) MODE1(SPI_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO6 = MODE0(GPIO6) MODE1(SPI_MISO) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() PD +GPIO7 = MODE0(GPIO7) MODE1(AUD_CLK) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN0) MODE7(TEST_OUT0) PD +GPIO8 = MODE0(GPIO8) MODE1(AUD_MOSI) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN1) MODE7(TEST_OUT1) PD +GPIO9 = MODE0(GPIO9) MODE1(AUD_MISO) MODE2() MODE3() MODE4() MODE5() MODE6(TEST_IN2) MODE7(TEST_OUT2) PD +GPIO10 = MODE0(GPIO10) MODE1(SIM1_AP_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO11 = MODE0(GPIO11) MODE1(SIM1_AP_SRST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO12 = MODE0(GPIO12) MODE1(SIM2_AP_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO13 = MODE0(GPIO13) MODE1(SIM2_AP_SRST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO14 = MODE0(GPIO14) MODE1(SIMLS1_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO15 = MODE0(GPIO15) MODE1(SIMLS1_SRST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO16 = MODE0(GPIO16) MODE1(SIMLS2_SCLK) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- +GPIO17 = MODE0(GPIO17) MODE1(SIMLS2_SRST) MODE2() MODE3() MODE4() MODE5() MODE6() MODE7() -- + + +[GPO] + +[EINT] +EINT_COUNT = 169 +EINT_DEBOUNCE_TIME_COUNT = 12 + + +[EINT_EX] +EINT_EXT_COUNT = 0 +EINT_EXT_DEBOUNCE_TIME_COUNT = 12 + + +[ADC] +ADC_COUNT = 6 + +[ADC_EX_PIN] + +[KEYPAD] +KEY_ROW = 4 +KEY_COLUMN = 4 + + +[POWER] +VDD_EMI +VDD1 +VCCK_VPROC +VCCK +DVDD28_SPDIF +DVDD28_NOR +DVDD28_MSDC2 +DVDD28_MSDC1 +DVDD28_MSDC0 +DVDD28_I2S +DVDD28_DPI +DVDD18_MIPITX +DVDD18_IO_WBCT +DVDD18_IO_SPDIF +DVDD18_IO_NOR +DVDD18_IO_MSDC2 +DVDD18_IO_MSDC1 +DVDD18_IO_MSDC0E +DVDD18_IO_MSDC0 +DVDD18_IO_I2S +DVDD18_IO_DPI +DVDD18_IO +AVDD33_USB +AVDD33_GLOBAL +AVDD18_USB +AVDD18_PLLGP +AVDD18_MEMPLL +AVDD18_HDMITX +AVDD18_GLOBAL +AVDD18_DAC +AVDD18_AP +AVDD12_GLOBAL + +[VDD_EMI] +VM +[VDD1] +VIO18 +[VCCK_VPROC] +VPROC +[VCCK] +VPROC +[DVDD28_SPDIF] +VMCH +VIO18 +[DVDD28_NOR] +VMCH +VIO18 +[DVDD28_MSDC2] +VMCH +VIO18 +[DVDD28_MSDC1] +VMCH +VIO18 +[DVDD28_MSDC0] +VEMC_3V3 +VIO18 +[DVDD28_I2S] +VMCH +VIO18 +[DVDD28_DPI] +VMCH +VIO18 +[DVDD18_MIPITX] +VIO18 +[DVDD18_IO_WBCT] +VIO18 +[DVDD18_IO_SPDIF] +VIO18 +[DVDD18_IO_NOR] +VIO18 +[DVDD18_IO_MSDC2] +VIO18 +[DVDD18_IO_MSDC1] +VIO18 +[DVDD18_IO_MSDC0E] +VIO18 +[DVDD18_IO_MSDC0] +VIO18 +[DVDD18_IO_I2S] +VIO18 +[DVDD18_IO_DPI] +VIO18 +[DVDD18_IO] +VIO18 +[AVDD33_USB] +VMC +VUSB +[AVDD33_GLOBAL] +VCAM_AF +[AVDD18_USB] +VIO18 +[AVDD18_PLLGP] +VIO18 +[AVDD18_MEMPLL] +VIO18 +[AVDD18_HDMITX] +VIO18 +[AVDD18_GLOBAL] +VCAMD_IO +[AVDD18_DAC] +VIO18 +[AVDD18_AP] +VIO18 +[AVDD12_GLOBAL] +VCAMD \ No newline at end of file diff --git a/tools/dct/PMIC_MT6322+6333PMUMP.cmp b/tools/dct/PMIC_MT6322+6333PMUMP.cmp new file mode 100755 index 000000000..c1be85c96 --- /dev/null +++ b/tools/dct/PMIC_MT6322+6333PMUMP.cmp @@ -0,0 +1,263 @@ +[PMIC_TABLE] +NUM_LDO = 21 + + +[LDO_NAME1] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_ldo_enable(MT6322_POWER_LDO_VMC,KAL_TRUE); +[LDO_ENABLE_OFF1] +pmic_ldo_enable(MT6322_POWER_LDO_VMC,KAL_FALSE); +[LDO_APPNAME1] +MT6322_POWER_LDO_VMC + +[LDO_NAME2] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_ldo_enable(MT6322_POWER_LDO_VMCH,KAL_TRUE); +[LDO_ENABLE_OFF2] +pmic_ldo_enable(MT6322_POWER_LDO_VMCH,KAL_FALSE); +[LDO_APPNAME2] +MT6322_POWER_LDO_VMCH + +[LDO_NAME3] +LDO_NAME=VEMC_3V3 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_ldo_enable(MT6322_POWER_LDO_VEMC_3V3,KAL_TRUE); +[LDO_ENABLE_OFF3] +pmic_ldo_enable(MT6322_POWER_LDO_VEMC_3V3,KAL_FALSE); +[LDO_APPNAME3] +MT6322_POWER_LDO_VEMC_3V3 + +[LDO_NAME4] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_ldo_enable(MT6322_POWER_LDO_VGP1,KAL_TRUE); +[LDO_ENABLE_OFF4] +pmic_ldo_enable(MT6322_POWER_LDO_VGP1,KAL_FALSE); +[LDO_APPNAME4] +MT6322_POWER_LDO_VGP1 + +[LDO_NAME5] +LDO_NAME=VGP2 +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_ldo_enable(MT6322_POWER_LDO_VGP2,KAL_TRUE); +[LDO_ENABLE_OFF5] +pmic_ldo_enable(MT6322_POWER_LDO_VGP2,KAL_FALSE); +[LDO_APPNAME5] +MT6322_POWER_LDO_VGP2 + +[LDO_NAME6] +LDO_NAME=VGP3 +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_ldo_enable(MT6322_POWER_LDO_VGP3,KAL_TRUE); +[LDO_ENABLE_OFF6] +pmic_ldo_enable(MT6322_POWER_LDO_VGP3,KAL_FALSE); +[LDO_APPNAME6] +MT6322_POWER_LDO_VGP3 + +[LDO_NAME7] +LDO_NAME=VCN_1V8 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_ldo_enable(MT6322_POWER_LDO_VCN_1V8,KAL_TRUE); +[LDO_ENABLE_OFF7] +pmic_ldo_enable(MT6322_POWER_LDO_VCN_1V8,KAL_FALSE); +[LDO_APPNAME7] +MT6322_POWER_LDO_VCN_1V8 + +[LDO_NAME8] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM1,KAL_TRUE); +[LDO_ENABLE_OFF8] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM1,KAL_FALSE); +[LDO_APPNAME8] +MT6322_POWER_LDO_VSIM1 + +[LDO_NAME9] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM2,KAL_TRUE); +[LDO_ENABLE_OFF9] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM2,KAL_FALSE); +[LDO_APPNAME9] +MT6322_POWER_LDO_VSIM2 + +[LDO_NAME10] +LDO_NAME=VCAM_AF +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_AF,KAL_TRUE); +[LDO_ENABLE_OFF10] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_AF,KAL_FALSE); +[LDO_APPNAME10] +MT6322_POWER_LDO_VCAM_AF + +[LDO_NAME11] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_ldo_enable(MT6322_POWER_LDO_VIBR,KAL_TRUE); +[LDO_ENABLE_OFF11] +pmic_ldo_enable(MT6322_POWER_LDO_VIBR,KAL_FALSE); +[LDO_APPNAME11] +MT6322_POWER_LDO_VIBR + +[LDO_NAME12] +LDO_NAME=VM +[LDO_ENABLE_CONFIGURABLE12] +CONFIGURABLE = YES +[LDO_ENABLE_ON12] +pmic_ldo_enable(MT6322_POWER_LDO_VM,KAL_TRUE); +[LDO_ENABLE_OFF12] +pmic_ldo_enable(MT6322_POWER_LDO_VM,KAL_FALSE); +[LDO_APPNAME12] +MT6322_POWER_LDO_VM + +[LDO_NAME13] +LDO_NAME=VRF18 +[LDO_ENABLE_CONFIGURABLE13] +CONFIGURABLE = YES +[LDO_ENABLE_ON13] +pmic_ldo_enable(MT6322_POWER_LDO_VRF18,KAL_TRUE); +[LDO_ENABLE_OFF13] +pmic_ldo_enable(MT6322_POWER_LDO_VRF18,KAL_FALSE); +[LDO_APPNAME13] +MT6322_POWER_LDO_VRF18 + +[LDO_NAME14] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE14] +CONFIGURABLE = YES +[LDO_ENABLE_ON14] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMD,KAL_TRUE); +[LDO_ENABLE_OFF14] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMD,KAL_FALSE); +[LDO_APPNAME14] +MT6322_POWER_LDO_VCAMD + +[LDO_NAME15] +LDO_NAME=VCAN_IO +[LDO_ENABLE_CONFIGURABLE15] +CONFIGURABLE = YES +[LDO_ENABLE_ON15] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_IO,KAL_TRUE); +[LDO_ENABLE_OFF15] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_IO,KAL_FALSE); +[LDO_APPNAME15] +MT6322_POWER_LDO_VCAM_IO + +[LDO_NAME16] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE16] +CONFIGURABLE = YES +[LDO_ENABLE_ON16] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMA,KAL_TRUE); +[LDO_ENABLE_OFF16] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMA,KAL_FALSE); +[LDO_APPNAME16] +MT6322_POWER_LDO_VCAMA + +[LDO_NAME17] +LDO_NAME=VCN33 +[LDO_ENABLE_CONFIGURABLE17] +CONFIGURABLE = YES +[LDO_ENABLE_ON17] +pmic_ldo_enable(MT6322_POWER_LDO_VCN33_WIFI,KAL_TRUE); +[LDO_ENABLE_OFF17] +pmic_ldo_enable(MT6322_POWER_LDO_VCN33_WIFI,KAL_FALSE); +[LDO_APPNAME17] +MT6322_POWER_LDO_VCN33_WIFI + +[LDO_NAME18] +LDO_NAME=VCN28 +[LDO_ENABLE_CONFIGURABLE18] +CONFIGURABLE = YES +[LDO_ENABLE_ON18] +pmic_ldo_enable(MT6322_POWER_LDO_VCN28,KAL_TRUE); +[LDO_ENABLE_OFF18] +pmic_ldo_enable(MT6322_POWER_LDO_VCN28,KAL_FALSE); +[LDO_APPNAME18] +MT6322_POWER_LDO_VCN28 + +[LDO_NAME19] +LDO_NAME=MT6333_VCORE +[LDO_ENABLE_CONFIGURABLE19] +CONFIGURABLE = YES +[LDO_ENABLE_ON19] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vcore_en(1); +#endif +[LDO_ENABLE_OFF19] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vcore_en(0); +#endif +[LDO_APPNAME19] +MT6322_POWER_LDO_DEFAULT + +[LDO_NAME20] +LDO_NAME=MT6333_VMEM +[LDO_ENABLE_CONFIGURABLE20] +CONFIGURABLE = YES +[LDO_ENABLE_ON20] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vmem_en(1); +#endif +[LDO_ENABLE_OFF20] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vmem_en(0); +#endif +[LDO_APPNAME20] +MT6322_POWER_LDO_DEFAULT + +[LDO_NAME21] +LDO_NAME=MT6333_VRF18 +[LDO_ENABLE_CONFIGURABLE21] +CONFIGURABLE = YES +[LDO_ENABLE_ON21] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vrf18_en(1); +#endif +[LDO_ENABLE_OFF21] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vrf18_en(0); +#endif +[LDO_APPNAME21] +MT6322_POWER_LDO_DEFAULT + +[LDO_APPNAME_DEFAULT] +MT6322_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include +#ifdef MTK_MT6333_SUPPORT +#include +#endif + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6322PMUMP.cmp b/tools/dct/PMIC_MT6322PMUMP.cmp new file mode 100755 index 000000000..3fe4e0beb --- /dev/null +++ b/tools/dct/PMIC_MT6322PMUMP.cmp @@ -0,0 +1,216 @@ +[PMIC_TABLE] +NUM_LDO = 18 + + +[LDO_NAME1] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_ldo_enable(MT6322_POWER_LDO_VMC,KAL_TRUE); +[LDO_ENABLE_OFF1] +pmic_ldo_enable(MT6322_POWER_LDO_VMC,KAL_FALSE); +[LDO_APPNAME1] +MT6322_POWER_LDO_VMC + +[LDO_NAME2] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_ldo_enable(MT6322_POWER_LDO_VMCH,KAL_TRUE); +[LDO_ENABLE_OFF2] +pmic_ldo_enable(MT6322_POWER_LDO_VMCH,KAL_FALSE); +[LDO_APPNAME2] +MT6322_POWER_LDO_VMCH + +[LDO_NAME3] +LDO_NAME=VEMC_3V3 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_ldo_enable(MT6322_POWER_LDO_VEMC_3V3,KAL_TRUE); +[LDO_ENABLE_OFF3] +pmic_ldo_enable(MT6322_POWER_LDO_VEMC_3V3,KAL_FALSE); +[LDO_APPNAME3] +MT6322_POWER_LDO_VEMC_3V3 + +[LDO_NAME4] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_ldo_enable(MT6322_POWER_LDO_VGP1,KAL_TRUE); +[LDO_ENABLE_OFF4] +pmic_ldo_enable(MT6322_POWER_LDO_VGP1,KAL_FALSE); +[LDO_APPNAME4] +MT6322_POWER_LDO_VGP1 + +[LDO_NAME5] +LDO_NAME=VGP2 +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_ldo_enable(MT6322_POWER_LDO_VGP2,KAL_TRUE); +[LDO_ENABLE_OFF5] +pmic_ldo_enable(MT6322_POWER_LDO_VGP2,KAL_FALSE); +[LDO_APPNAME5] +MT6322_POWER_LDO_VGP2 + +[LDO_NAME6] +LDO_NAME=VGP3 +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_ldo_enable(MT6322_POWER_LDO_VGP3,KAL_TRUE); +[LDO_ENABLE_OFF6] +pmic_ldo_enable(MT6322_POWER_LDO_VGP3,KAL_FALSE); +[LDO_APPNAME6] +MT6322_POWER_LDO_VGP3 + +[LDO_NAME7] +LDO_NAME=VCN_1V8 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_ldo_enable(MT6322_POWER_LDO_VCN_1V8,KAL_TRUE); +[LDO_ENABLE_OFF7] +pmic_ldo_enable(MT6322_POWER_LDO_VCN_1V8,KAL_FALSE); +[LDO_APPNAME7] +MT6322_POWER_LDO_VCN_1V8 + +[LDO_NAME8] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM1,KAL_TRUE); +[LDO_ENABLE_OFF8] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM1,KAL_FALSE); +[LDO_APPNAME8] +MT6322_POWER_LDO_VSIM1 + +[LDO_NAME9] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM2,KAL_TRUE); +[LDO_ENABLE_OFF9] +pmic_ldo_enable(MT6322_POWER_LDO_VSIM2,KAL_FALSE); +[LDO_APPNAME9] +MT6322_POWER_LDO_VSIM2 + +[LDO_NAME10] +LDO_NAME=VCAM_AF +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_AF,KAL_TRUE); +[LDO_ENABLE_OFF10] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_AF,KAL_FALSE); +[LDO_APPNAME10] +MT6322_POWER_LDO_VCAM_AF + +[LDO_NAME11] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_ldo_enable(MT6322_POWER_LDO_VIBR,KAL_TRUE); +[LDO_ENABLE_OFF11] +pmic_ldo_enable(MT6322_POWER_LDO_VIBR,KAL_FALSE); +[LDO_APPNAME11] +MT6322_POWER_LDO_VIBR + +[LDO_NAME12] +LDO_NAME=VM +[LDO_ENABLE_CONFIGURABLE12] +CONFIGURABLE = YES +[LDO_ENABLE_ON12] +pmic_ldo_enable(MT6322_POWER_LDO_VM,KAL_TRUE); +[LDO_ENABLE_OFF12] +pmic_ldo_enable(MT6322_POWER_LDO_VM,KAL_FALSE); +[LDO_APPNAME12] +MT6322_POWER_LDO_VM + +[LDO_NAME13] +LDO_NAME=VRF18 +[LDO_ENABLE_CONFIGURABLE13] +CONFIGURABLE = YES +[LDO_ENABLE_ON13] +pmic_ldo_enable(MT6322_POWER_LDO_VRF18,KAL_TRUE); +[LDO_ENABLE_OFF13] +pmic_ldo_enable(MT6322_POWER_LDO_VRF18,KAL_FALSE); +[LDO_APPNAME13] +MT6322_POWER_LDO_VRF18 + +[LDO_NAME14] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE14] +CONFIGURABLE = YES +[LDO_ENABLE_ON14] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMD,KAL_TRUE); +[LDO_ENABLE_OFF14] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMD,KAL_FALSE); +[LDO_APPNAME14] +MT6322_POWER_LDO_VCAMD + +[LDO_NAME15] +LDO_NAME=VCAN_IO +[LDO_ENABLE_CONFIGURABLE15] +CONFIGURABLE = YES +[LDO_ENABLE_ON15] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_IO,KAL_TRUE); +[LDO_ENABLE_OFF15] +pmic_ldo_enable(MT6322_POWER_LDO_VCAM_IO,KAL_FALSE); +[LDO_APPNAME15] +MT6322_POWER_LDO_VCAM_IO + +[LDO_NAME16] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE16] +CONFIGURABLE = YES +[LDO_ENABLE_ON16] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMA,KAL_TRUE); +[LDO_ENABLE_OFF16] +pmic_ldo_enable(MT6322_POWER_LDO_VCAMA,KAL_FALSE); +[LDO_APPNAME16] +MT6322_POWER_LDO_VCAMA + +[LDO_NAME17] +LDO_NAME=VCN33 +[LDO_ENABLE_CONFIGURABLE17] +CONFIGURABLE = YES +[LDO_ENABLE_ON17] +pmic_ldo_enable(MT6322_POWER_LDO_VCN33_WIFI,KAL_TRUE); +[LDO_ENABLE_OFF17] +pmic_ldo_enable(MT6322_POWER_LDO_VCN33_WIFI,KAL_FALSE); +[LDO_APPNAME17] +MT6322_POWER_LDO_VCN33_WIFI + +[LDO_NAME18] +LDO_NAME=VCN28 +[LDO_ENABLE_CONFIGURABLE18] +CONFIGURABLE = YES +[LDO_ENABLE_ON18] +pmic_ldo_enable(MT6322_POWER_LDO_VCN28,KAL_TRUE); +[LDO_ENABLE_OFF18] +pmic_ldo_enable(MT6322_POWER_LDO_VCN28,KAL_FALSE); +[LDO_APPNAME18] +MT6322_POWER_LDO_VCN28 + + +[LDO_APPNAME_DEFAULT] +MT6322_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6323+6333PMUMP.cmp b/tools/dct/PMIC_MT6323+6333PMUMP.cmp new file mode 100755 index 000000000..c8eb829ef --- /dev/null +++ b/tools/dct/PMIC_MT6323+6333PMUMP.cmp @@ -0,0 +1,263 @@ +[PMIC_TABLE] +NUM_LDO = 21 + + +[LDO_NAME1] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_ldo_enable(MT6323_POWER_LDO_VMC,KAL_TRUE); +[LDO_ENABLE_OFF1] +pmic_ldo_enable(MT6323_POWER_LDO_VMC,KAL_FALSE); +[LDO_APPNAME1] +MT6323_POWER_LDO_VMC + +[LDO_NAME2] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_ldo_enable(MT6323_POWER_LDO_VMCH,KAL_TRUE); +[LDO_ENABLE_OFF2] +pmic_ldo_enable(MT6323_POWER_LDO_VMCH,KAL_FALSE); +[LDO_APPNAME2] +MT6323_POWER_LDO_VMCH + +[LDO_NAME3] +LDO_NAME=VEMC_3V3 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_ldo_enable(MT6323_POWER_LDO_VEMC_3V3,KAL_TRUE); +[LDO_ENABLE_OFF3] +pmic_ldo_enable(MT6323_POWER_LDO_VEMC_3V3,KAL_FALSE); +[LDO_APPNAME3] +MT6323_POWER_LDO_VEMC_3V3 + +[LDO_NAME4] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_ldo_enable(MT6323_POWER_LDO_VGP1,KAL_TRUE); +[LDO_ENABLE_OFF4] +pmic_ldo_enable(MT6323_POWER_LDO_VGP1,KAL_FALSE); +[LDO_APPNAME4] +MT6323_POWER_LDO_VGP1 + +[LDO_NAME5] +LDO_NAME=VGP2 +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_ldo_enable(MT6323_POWER_LDO_VGP2,KAL_TRUE); +[LDO_ENABLE_OFF5] +pmic_ldo_enable(MT6323_POWER_LDO_VGP2,KAL_FALSE); +[LDO_APPNAME5] +MT6323_POWER_LDO_VGP2 + +[LDO_NAME6] +LDO_NAME=VGP3 +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_ldo_enable(MT6323_POWER_LDO_VGP3,KAL_TRUE); +[LDO_ENABLE_OFF6] +pmic_ldo_enable(MT6323_POWER_LDO_VGP3,KAL_FALSE); +[LDO_APPNAME6] +MT6323_POWER_LDO_VGP3 + +[LDO_NAME7] +LDO_NAME=VCN_1V8 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_ldo_enable(MT6323_POWER_LDO_VCN_1V8,KAL_TRUE); +[LDO_ENABLE_OFF7] +pmic_ldo_enable(MT6323_POWER_LDO_VCN_1V8,KAL_FALSE); +[LDO_APPNAME7] +MT6323_POWER_LDO_VCN_1V8 + +[LDO_NAME8] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM1,KAL_TRUE); +[LDO_ENABLE_OFF8] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM1,KAL_FALSE); +[LDO_APPNAME8] +MT6323_POWER_LDO_VSIM1 + +[LDO_NAME9] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM2,KAL_TRUE); +[LDO_ENABLE_OFF9] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM2,KAL_FALSE); +[LDO_APPNAME9] +MT6323_POWER_LDO_VSIM2 + +[LDO_NAME10] +LDO_NAME=VCAM_AF +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_AF,KAL_TRUE); +[LDO_ENABLE_OFF10] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_AF,KAL_FALSE); +[LDO_APPNAME10] +MT6323_POWER_LDO_VCAM_AF + +[LDO_NAME11] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_ldo_enable(MT6323_POWER_LDO_VIBR,KAL_TRUE); +[LDO_ENABLE_OFF11] +pmic_ldo_enable(MT6323_POWER_LDO_VIBR,KAL_FALSE); +[LDO_APPNAME11] +MT6323_POWER_LDO_VIBR + +[LDO_NAME12] +LDO_NAME=VM +[LDO_ENABLE_CONFIGURABLE12] +CONFIGURABLE = YES +[LDO_ENABLE_ON12] +pmic_ldo_enable(MT6323_POWER_LDO_VM,KAL_TRUE); +[LDO_ENABLE_OFF12] +pmic_ldo_enable(MT6323_POWER_LDO_VM,KAL_FALSE); +[LDO_APPNAME12] +MT6323_POWER_LDO_VM + +[LDO_NAME13] +LDO_NAME=VRF18 +[LDO_ENABLE_CONFIGURABLE13] +CONFIGURABLE = YES +[LDO_ENABLE_ON13] +pmic_ldo_enable(MT6323_POWER_LDO_VRF18,KAL_TRUE); +[LDO_ENABLE_OFF13] +pmic_ldo_enable(MT6323_POWER_LDO_VRF18,KAL_FALSE); +[LDO_APPNAME13] +MT6323_POWER_LDO_VRF18 + +[LDO_NAME14] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE14] +CONFIGURABLE = YES +[LDO_ENABLE_ON14] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMD,KAL_TRUE); +[LDO_ENABLE_OFF14] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMD,KAL_FALSE); +[LDO_APPNAME14] +MT6323_POWER_LDO_VCAMD + +[LDO_NAME15] +LDO_NAME=VCAN_IO +[LDO_ENABLE_CONFIGURABLE15] +CONFIGURABLE = YES +[LDO_ENABLE_ON15] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_IO,KAL_TRUE); +[LDO_ENABLE_OFF15] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_IO,KAL_FALSE); +[LDO_APPNAME15] +MT6323_POWER_LDO_VCAM_IO + +[LDO_NAME16] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE16] +CONFIGURABLE = YES +[LDO_ENABLE_ON16] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMA,KAL_TRUE); +[LDO_ENABLE_OFF16] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMA,KAL_FALSE); +[LDO_APPNAME16] +MT6323_POWER_LDO_VCAMA + +[LDO_NAME17] +LDO_NAME=VCN33 +[LDO_ENABLE_CONFIGURABLE17] +CONFIGURABLE = YES +[LDO_ENABLE_ON17] +pmic_ldo_enable(MT6323_POWER_LDO_VCN33_WIFI,KAL_TRUE); +[LDO_ENABLE_OFF17] +pmic_ldo_enable(MT6323_POWER_LDO_VCN33_WIFI,KAL_FALSE); +[LDO_APPNAME17] +MT6323_POWER_LDO_VCN33_WIFI + +[LDO_NAME18] +LDO_NAME=VCN28 +[LDO_ENABLE_CONFIGURABLE18] +CONFIGURABLE = YES +[LDO_ENABLE_ON18] +pmic_ldo_enable(MT6323_POWER_LDO_VCN28,KAL_TRUE); +[LDO_ENABLE_OFF18] +pmic_ldo_enable(MT6323_POWER_LDO_VCN28,KAL_FALSE); +[LDO_APPNAME18] +MT6323_POWER_LDO_VCN28 + +[LDO_NAME19] +LDO_NAME=MT6333_VCORE +[LDO_ENABLE_CONFIGURABLE19] +CONFIGURABLE = YES +[LDO_ENABLE_ON19] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vcore_en(1); +#endif +[LDO_ENABLE_OFF19] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vcore_en(0); +#endif +[LDO_APPNAME19] +MT6323_POWER_LDO_DEFAULT + +[LDO_NAME20] +LDO_NAME=MT6333_VMEM +[LDO_ENABLE_CONFIGURABLE20] +CONFIGURABLE = YES +[LDO_ENABLE_ON20] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vmem_en(1); +#endif +[LDO_ENABLE_OFF20] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vmem_en(0); +#endif +[LDO_APPNAME20] +MT6323_POWER_LDO_DEFAULT + +[LDO_NAME21] +LDO_NAME=MT6333_VRF18 +[LDO_ENABLE_CONFIGURABLE21] +CONFIGURABLE = YES +[LDO_ENABLE_ON21] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vrf18_en(1); +#endif +[LDO_ENABLE_OFF21] +#ifdef MTK_MT6333_SUPPORT +mt6333_set_vrf18_en(0); +#endif +[LDO_APPNAME21] +MT6323_POWER_LDO_DEFAULT + +[LDO_APPNAME_DEFAULT] +MT6323_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include +#ifdef MTK_MT6333_SUPPORT +#include +#endif + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6323PMUMP.cmp b/tools/dct/PMIC_MT6323PMUMP.cmp new file mode 100755 index 000000000..e600e6df3 --- /dev/null +++ b/tools/dct/PMIC_MT6323PMUMP.cmp @@ -0,0 +1,216 @@ +[PMIC_TABLE] +NUM_LDO = 18 + + +[LDO_NAME1] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_ldo_enable(MT6323_POWER_LDO_VMC,KAL_TRUE); +[LDO_ENABLE_OFF1] +pmic_ldo_enable(MT6323_POWER_LDO_VMC,KAL_FALSE); +[LDO_APPNAME1] +MT6323_POWER_LDO_VMC + +[LDO_NAME2] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_ldo_enable(MT6323_POWER_LDO_VMCH,KAL_TRUE); +[LDO_ENABLE_OFF2] +pmic_ldo_enable(MT6323_POWER_LDO_VMCH,KAL_FALSE); +[LDO_APPNAME2] +MT6323_POWER_LDO_VMCH + +[LDO_NAME3] +LDO_NAME=VEMC_3V3 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_ldo_enable(MT6323_POWER_LDO_VEMC_3V3,KAL_TRUE); +[LDO_ENABLE_OFF3] +pmic_ldo_enable(MT6323_POWER_LDO_VEMC_3V3,KAL_FALSE); +[LDO_APPNAME3] +MT6323_POWER_LDO_VEMC_3V3 + +[LDO_NAME4] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_ldo_enable(MT6323_POWER_LDO_VGP1,KAL_TRUE); +[LDO_ENABLE_OFF4] +pmic_ldo_enable(MT6323_POWER_LDO_VGP1,KAL_FALSE); +[LDO_APPNAME4] +MT6323_POWER_LDO_VGP1 + +[LDO_NAME5] +LDO_NAME=VGP2 +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_ldo_enable(MT6323_POWER_LDO_VGP2,KAL_TRUE); +[LDO_ENABLE_OFF5] +pmic_ldo_enable(MT6323_POWER_LDO_VGP2,KAL_FALSE); +[LDO_APPNAME5] +MT6323_POWER_LDO_VGP2 + +[LDO_NAME6] +LDO_NAME=VGP3 +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_ldo_enable(MT6323_POWER_LDO_VGP3,KAL_TRUE); +[LDO_ENABLE_OFF6] +pmic_ldo_enable(MT6323_POWER_LDO_VGP3,KAL_FALSE); +[LDO_APPNAME6] +MT6323_POWER_LDO_VGP3 + +[LDO_NAME7] +LDO_NAME=VCN_1V8 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_ldo_enable(MT6323_POWER_LDO_VCN_1V8,KAL_TRUE); +[LDO_ENABLE_OFF7] +pmic_ldo_enable(MT6323_POWER_LDO_VCN_1V8,KAL_FALSE); +[LDO_APPNAME7] +MT6323_POWER_LDO_VCN_1V8 + +[LDO_NAME8] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM1,KAL_TRUE); +[LDO_ENABLE_OFF8] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM1,KAL_FALSE); +[LDO_APPNAME8] +MT6323_POWER_LDO_VSIM1 + +[LDO_NAME9] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM2,KAL_TRUE); +[LDO_ENABLE_OFF9] +pmic_ldo_enable(MT6323_POWER_LDO_VSIM2,KAL_FALSE); +[LDO_APPNAME9] +MT6323_POWER_LDO_VSIM2 + +[LDO_NAME10] +LDO_NAME=VCAM_AF +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_AF,KAL_TRUE); +[LDO_ENABLE_OFF10] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_AF,KAL_FALSE); +[LDO_APPNAME10] +MT6323_POWER_LDO_VCAM_AF + +[LDO_NAME11] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_ldo_enable(MT6323_POWER_LDO_VIBR,KAL_TRUE); +[LDO_ENABLE_OFF11] +pmic_ldo_enable(MT6323_POWER_LDO_VIBR,KAL_FALSE); +[LDO_APPNAME11] +MT6323_POWER_LDO_VIBR + +[LDO_NAME12] +LDO_NAME=VM +[LDO_ENABLE_CONFIGURABLE12] +CONFIGURABLE = YES +[LDO_ENABLE_ON12] +pmic_ldo_enable(MT6323_POWER_LDO_VM,KAL_TRUE); +[LDO_ENABLE_OFF12] +pmic_ldo_enable(MT6323_POWER_LDO_VM,KAL_FALSE); +[LDO_APPNAME12] +MT6323_POWER_LDO_VM + +[LDO_NAME13] +LDO_NAME=VRF18 +[LDO_ENABLE_CONFIGURABLE13] +CONFIGURABLE = YES +[LDO_ENABLE_ON13] +pmic_ldo_enable(MT6323_POWER_LDO_VRF18,KAL_TRUE); +[LDO_ENABLE_OFF13] +pmic_ldo_enable(MT6323_POWER_LDO_VRF18,KAL_FALSE); +[LDO_APPNAME13] +MT6323_POWER_LDO_VRF18 + +[LDO_NAME14] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE14] +CONFIGURABLE = YES +[LDO_ENABLE_ON14] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMD,KAL_TRUE); +[LDO_ENABLE_OFF14] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMD,KAL_FALSE); +[LDO_APPNAME14] +MT6323_POWER_LDO_VCAMD + +[LDO_NAME15] +LDO_NAME=VCAN_IO +[LDO_ENABLE_CONFIGURABLE15] +CONFIGURABLE = YES +[LDO_ENABLE_ON15] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_IO,KAL_TRUE); +[LDO_ENABLE_OFF15] +pmic_ldo_enable(MT6323_POWER_LDO_VCAM_IO,KAL_FALSE); +[LDO_APPNAME15] +MT6323_POWER_LDO_VCAM_IO + +[LDO_NAME16] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE16] +CONFIGURABLE = YES +[LDO_ENABLE_ON16] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMA,KAL_TRUE); +[LDO_ENABLE_OFF16] +pmic_ldo_enable(MT6323_POWER_LDO_VCAMA,KAL_FALSE); +[LDO_APPNAME16] +MT6323_POWER_LDO_VCAMA + +[LDO_NAME17] +LDO_NAME=VCN33 +[LDO_ENABLE_CONFIGURABLE17] +CONFIGURABLE = YES +[LDO_ENABLE_ON17] +pmic_ldo_enable(MT6323_POWER_LDO_VCN33_WIFI,KAL_TRUE); +[LDO_ENABLE_OFF17] +pmic_ldo_enable(MT6323_POWER_LDO_VCN33_WIFI,KAL_FALSE); +[LDO_APPNAME17] +MT6323_POWER_LDO_VCN33_WIFI + +[LDO_NAME18] +LDO_NAME=VCN28 +[LDO_ENABLE_CONFIGURABLE18] +CONFIGURABLE = YES +[LDO_ENABLE_ON18] +pmic_ldo_enable(MT6323_POWER_LDO_VCN28,KAL_TRUE); +[LDO_ENABLE_OFF18] +pmic_ldo_enable(MT6323_POWER_LDO_VCN28,KAL_FALSE); +[LDO_APPNAME18] +MT6323_POWER_LDO_VCN28 + + +[LDO_APPNAME_DEFAULT] +MT6323_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6325PMUMP.cmp b/tools/dct/PMIC_MT6325PMUMP.cmp new file mode 100755 index 000000000..3f7001205 --- /dev/null +++ b/tools/dct/PMIC_MT6325PMUMP.cmp @@ -0,0 +1,325 @@ +[PMIC_TABLE] +NUM_LDO = 28 + + +[LDO_NAME1] +LDO_NAME=VTCXO0 +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_ldo_enable(MT6325_POWER_LDO_VTCXO0,KAL_TRUE); +[LDO_ENABLE_OFF1] +pmic_ldo_enable(MT6325_POWER_LDO_VTCXO0,KAL_FALSE); +[LDO_APPNAME1] +MT6325_POWER_LDO_VTCXO0 + +[LDO_NAME2] +LDO_NAME=VTCXO1 +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_ldo_enable(MT6325_POWER_LDO_VTCXO1,KAL_TRUE); +[LDO_ENABLE_OFF2] +pmic_ldo_enable(MT6325_POWER_LDO_VTCXO1,KAL_FALSE); +[LDO_APPNAME2] +MT6325_POWER_LDO_VTCXO1 + +[LDO_NAME3] +LDO_NAME=VAUD28 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_ldo_enable(MT6325_POWER_LDO_VAUD28,KAL_TRUE); +[LDO_ENABLE_OFF3] +pmic_ldo_enable(MT6325_POWER_LDO_VAUD28,KAL_FALSE); +[LDO_APPNAME3] +MT6325_POWER_LDO_VAUD28 + +[LDO_NAME4] +LDO_NAME=VAUXA28 +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_ldo_enable(MT6325_POWER_LDO_VAUXA28,KAL_TRUE); +[LDO_ENABLE_OFF4] +pmic_ldo_enable(MT6325_POWER_LDO_VAUXA28,KAL_FALSE); +[LDO_APPNAME4] +MT6325_POWER_LDO_VAUXA28 + +[LDO_NAME5] +LDO_NAME=VBIF28 +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_ldo_enable(MT6325_POWER_LDO_VBIF28,KAL_TRUE); +[LDO_ENABLE_OFF5] +pmic_ldo_enable(MT6325_POWER_LDO_VBIF28,KAL_FALSE); +[LDO_APPNAME5] +MT6325_POWER_LDO_VBIF28 + +[LDO_NAME6] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_ldo_enable(MT6325_POWER_LDO_VCAMA,KAL_TRUE); +[LDO_ENABLE_OFF6] +pmic_ldo_enable(MT6325_POWER_LDO_VCAMA,KAL_FALSE); +[LDO_APPNAME6] +MT6325_POWER_LDO_VCAMA + +[LDO_NAME7] +LDO_NAME=VCN28 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_ldo_enable(MT6325_POWER_LDO_VCN28,KAL_TRUE); +[LDO_ENABLE_OFF7] +pmic_ldo_enable(MT6325_POWER_LDO_VCN28,KAL_FALSE); +[LDO_APPNAME7] +MT6325_POWER_LDO_VCN28 + +[LDO_NAME8] +LDO_NAME=VCN33 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_ldo_enable(MT6325_POWER_LDO_VCN33,KAL_TRUE); +[LDO_ENABLE_OFF8] +pmic_ldo_enable(MT6325_POWER_LDO_VCN33,KAL_FALSE); +[LDO_APPNAME8] +MT6325_POWER_LDO_VCN33 + +[LDO_NAME9] +LDO_NAME=VRF18_1 +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_ldo_enable(MT6325_POWER_LDO_VRF18_1,KAL_TRUE); +[LDO_ENABLE_OFF9] +pmic_ldo_enable(MT6325_POWER_LDO_VRF18_1,KAL_FALSE); +[LDO_APPNAME9] +MT6325_POWER_LDO_VRF18_1 + +[LDO_NAME10] +LDO_NAME=VUSB33 +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_ldo_enable(MT6325_POWER_LDO_VUSB33,KAL_TRUE); +[LDO_ENABLE_OFF10] +pmic_ldo_enable(MT6325_POWER_LDO_VUSB33,KAL_FALSE); +[LDO_APPNAME10] +MT6325_POWER_LDO_VUSB33 + +[LDO_NAME11] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_ldo_enable(MT6325_POWER_LDO_VMCH,KAL_TRUE); +[LDO_ENABLE_OFF11] +pmic_ldo_enable(MT6325_POWER_LDO_VMCH,KAL_FALSE); +[LDO_APPNAME11] +MT6325_POWER_LDO_VMCH + +[LDO_NAME12] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE12] +CONFIGURABLE = YES +[LDO_ENABLE_ON12] +pmic_ldo_enable(MT6325_POWER_LDO_VMC,KAL_TRUE); +[LDO_ENABLE_OFF12] +pmic_ldo_enable(MT6325_POWER_LDO_VMC,KAL_FALSE); +[LDO_APPNAME12] +MT6325_POWER_LDO_VMC + +[LDO_NAME13] +LDO_NAME=VEMC33 +[LDO_ENABLE_CONFIGURABLE13] +CONFIGURABLE = YES +[LDO_ENABLE_ON13] +pmic_ldo_enable(MT6325_POWER_LDO_VEMC33,KAL_TRUE); +[LDO_ENABLE_OFF13] +pmic_ldo_enable(MT6325_POWER_LDO_VEMC33,KAL_FALSE); +[LDO_APPNAME13] +MT6325_POWER_LDO_VEMC33 + +[LDO_NAME14] +LDO_NAME=VIO28 +[LDO_ENABLE_CONFIGURABLE14] +CONFIGURABLE = YES +[LDO_ENABLE_ON14] +pmic_ldo_enable(MT6325_POWER_LDO_VIO28,KAL_TRUE); +[LDO_ENABLE_OFF14] +pmic_ldo_enable(MT6325_POWER_LDO_VIO28,KAL_FALSE); +[LDO_APPNAME14] +MT6325_POWER_LDO_VIO28 + +[LDO_NAME15] +LDO_NAME=VCAM_AF +[LDO_ENABLE_CONFIGURABLE15] +CONFIGURABLE = YES +[LDO_ENABLE_ON15] +pmic_ldo_enable(MT6325_POWER_LDO_VCAM_AF,KAL_TRUE); +[LDO_ENABLE_OFF15] +pmic_ldo_enable(MT6325_POWER_LDO_VCAM_AF,KAL_FALSE); +[LDO_APPNAME15] +MT6325_POWER_LDO_VCAM_AF + +[LDO_NAME16] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE16] +CONFIGURABLE = YES +[LDO_ENABLE_ON16] +pmic_ldo_enable(MT6325_POWER_LDO_VGP1,KAL_TRUE); +[LDO_ENABLE_OFF16] +pmic_ldo_enable(MT6325_POWER_LDO_VGP1,KAL_FALSE); +[LDO_APPNAME16] +MT6325_POWER_LDO_VGP1 + +[LDO_NAME17] +LDO_NAME=VEFUSE +[LDO_ENABLE_CONFIGURABLE17] +CONFIGURABLE = YES +[LDO_ENABLE_ON17] +pmic_ldo_enable(MT6325_POWER_LDO_VEFUSE,KAL_TRUE); +[LDO_ENABLE_OFF17] +pmic_ldo_enable(MT6325_POWER_LDO_VEFUSE,KAL_FALSE); +[LDO_APPNAME17] +MT6325_POWER_LDO_VEFUSE + +[LDO_NAME18] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE18] +CONFIGURABLE = YES +[LDO_ENABLE_ON18] +pmic_ldo_enable(MT6325_POWER_LDO_VSIM1,KAL_TRUE); +[LDO_ENABLE_OFF18] +pmic_ldo_enable(MT6325_POWER_LDO_VSIM1,KAL_FALSE); +[LDO_APPNAME18] +MT6325_POWER_LDO_VSIM1 + +[LDO_NAME19] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE19] +CONFIGURABLE = YES +[LDO_ENABLE_ON19] +pmic_ldo_enable(MT6325_POWER_LDO_VSIM2,KAL_TRUE); +[LDO_ENABLE_OFF19] +pmic_ldo_enable(MT6325_POWER_LDO_VSIM2,KAL_FALSE); +[LDO_APPNAME19] +MT6325_POWER_LDO_VSIM2 + +[LDO_NAME20] +LDO_NAME=VMIPI +[LDO_ENABLE_CONFIGURABLE20] +CONFIGURABLE = YES +[LDO_ENABLE_ON20] +pmic_ldo_enable(MT6325_POWER_LDO_VMIPI,KAL_TRUE); +[LDO_ENABLE_OFF20] +pmic_ldo_enable(MT6325_POWER_LDO_VMIPI,KAL_FALSE); +[LDO_APPNAME20] +MT6325_POWER_LDO_VMIPI + +[LDO_NAME21] +LDO_NAME=VCN18 +[LDO_ENABLE_CONFIGURABLE21] +CONFIGURABLE = YES +[LDO_ENABLE_ON21] +pmic_ldo_enable(MT6325_POWER_LDO_VCN18,KAL_TRUE); +[LDO_ENABLE_OFF21] +pmic_ldo_enable(MT6325_POWER_LDO_VCN18,KAL_FALSE); +[LDO_APPNAME21] +MT6325_POWER_LDO_VCN18 + +[LDO_NAME22] +LDO_NAME=VGP2 +[LDO_ENABLE_CONFIGURABLE22] +CONFIGURABLE = YES +[LDO_ENABLE_ON22] +pmic_ldo_enable(MT6325_POWER_LDO_VGP2,KAL_TRUE); +[LDO_ENABLE_OFF22] +pmic_ldo_enable(MT6325_POWER_LDO_VGP2,KAL_FALSE); +[LDO_APPNAME22] +MT6325_POWER_LDO_VGP2 + +[LDO_NAME23] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE23] +CONFIGURABLE = YES +[LDO_ENABLE_ON23] +pmic_ldo_enable(MT6325_POWER_LDO_VCAMD,KAL_TRUE); +[LDO_ENABLE_OFF23] +pmic_ldo_enable(MT6325_POWER_LDO_VCAMD,KAL_FALSE); +[LDO_APPNAME23] +MT6325_POWER_LDO_VCAMD + +[LDO_NAME24] +LDO_NAME=VCAM_IO +[LDO_ENABLE_CONFIGURABLE24] +CONFIGURABLE = YES +[LDO_ENABLE_ON24] +pmic_ldo_enable(MT6325_POWER_LDO_VCAM_IO,KAL_TRUE); +[LDO_ENABLE_OFF24] +pmic_ldo_enable(MT6325_POWER_LDO_VCAM_IO,KAL_FALSE); +[LDO_APPNAME24] +MT6325_POWER_LDO_VCAM_IO + +[LDO_NAME25] +LDO_NAME=VSRAM_DVFS1 +[LDO_ENABLE_CONFIGURABLE25] +CONFIGURABLE = YES +[LDO_ENABLE_ON25] +pmic_ldo_enable(MT6325_POWER_LDO_VSRAM_DVFS1,KAL_TRUE); +[LDO_ENABLE_OFF25] +pmic_ldo_enable(MT6325_POWER_LDO_VSRAM_DVFS1,KAL_FALSE); +[LDO_APPNAME25] +MT6325_POWER_LDO_VSRAM_DVFS1 + +[LDO_NAME26] +LDO_NAME=VGP3 +[LDO_ENABLE_CONFIGURABLE26] +CONFIGURABLE = YES +[LDO_ENABLE_ON26] +pmic_ldo_enable(MT6325_POWER_LDO_VGP3,KAL_TRUE); +[LDO_ENABLE_OFF26] +pmic_ldo_enable(MT6325_POWER_LDO_VGP3,KAL_FALSE); +[LDO_APPNAME26] +MT6325_POWER_LDO_VGP3 + +[LDO_NAME27] +LDO_NAME=VBIASN +[LDO_ENABLE_CONFIGURABLE27] +CONFIGURABLE = YES +[LDO_ENABLE_ON27] +pmic_ldo_enable(MT6325_POWER_LDO_VBIASN,KAL_TRUE); +[LDO_ENABLE_OFF27] +pmic_ldo_enable(MT6325_POWER_LDO_VBIASN,KAL_FALSE); +[LDO_APPNAME27] +MT6325_POWER_LDO_VBIASN + +[LDO_NAME28] +LDO_NAME=VRTC +[LDO_ENABLE_CONFIGURABLE28] +CONFIGURABLE = YES +[LDO_ENABLE_ON28] +pmic_ldo_enable(MT6325_POWER_LDO_VRTC,KAL_TRUE); +[LDO_ENABLE_OFF28] +pmic_ldo_enable(MT6325_POWER_LDO_VRTC,KAL_FALSE); +[LDO_APPNAME28] +MT6325_POWER_LDO_VRTC + +[LDO_APPNAME_DEFAULT] +MT65XX_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6328PMUMP.cmp b/tools/dct/PMIC_MT6328PMUMP.cmp new file mode 100755 index 000000000..ee2a416cc --- /dev/null +++ b/tools/dct/PMIC_MT6328PMUMP.cmp @@ -0,0 +1,108 @@ +[PMIC_TABLE] +NUM_LDO = 8 + + +[LDO_NAME1] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_set_register_value(PMIC_RG_VCAMA_EN,1); +[LDO_ENABLE_OFF1] +pmic_set_register_value(PMIC_RG_VCAMA_EN,0); +[LDO_APPNAME1] +MT6328_POWER_LDO_VCAMA + +[LDO_NAME2] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_set_register_value(PMIC_RG_VTCXO_1_EN,1); +[LDO_ENABLE_OFF2] +pmic_set_register_value(PMIC_RG_VTCXO_1_EN,0); +[LDO_APPNAME2] +MT6328_POWER_LDO_VSIM1 + +[LDO_NAME3] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_set_register_value(PMIC_RG_VSIM2_EN,1); +[LDO_ENABLE_OFF3] +pmic_set_register_value(PMIC_RG_VSIM2_EN,0); +[LDO_APPNAME3] +MT6328_POWER_LDO_VSIM2 + +[LDO_NAME4] +LDO_NAME=VCAM_AF +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_set_register_value(PMIC_RG_VCAMAF_EN,1); +[LDO_ENABLE_OFF4] +pmic_set_register_value(PMIC_RG_VCAMAF_EN,0); +[LDO_APPNAME4] +MT6328_POWER_LDO_VCAM_AF + +[LDO_NAME5] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_set_register_value(PMIC_RG_VGP1_EN,1); +[LDO_ENABLE_OFF5] +pmic_set_register_value(PMIC_RG_VGP1_EN,0); +[LDO_APPNAME5] +MT6328_POWER_LDO_VGP1 + +[LDO_NAME6] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_set_register_value(PMIC_RG_VIBR_EN,1); +[LDO_ENABLE_OFF6] +pmic_set_register_value(PMIC_RG_VIBR_EN,0); +[LDO_APPNAME6] +MT6328_POWER_LDO_VIBR + +[LDO_NAME7] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_set_register_value(PMIC_RG_VCAMD_EN,1); +[LDO_ENABLE_OFF7] +pmic_set_register_value(PMIC_RG_VCAMD_EN,0); +[LDO_APPNAME7] +MT6328_POWER_LDO_VCAMD + +[LDO_NAME8] +LDO_NAME=VCAM_IO +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_set_register_value(PMIC_RG_VCAMIO_EN,1); +[LDO_ENABLE_OFF8] +pmic_set_register_value(PMIC_RG_VCAMIO_EN,0); +[LDO_APPNAME8] +MT6328_POWER_LDO_VCAM_IO + + +[LDO_APPNAME_DEFAULT] +MT65XX_POWER_NONE + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6331+6332PMUMP.cmp b/tools/dct/PMIC_MT6331+6332PMUMP.cmp new file mode 100755 index 000000000..c4155955b --- /dev/null +++ b/tools/dct/PMIC_MT6331+6332PMUMP.cmp @@ -0,0 +1,358 @@ +[PMIC_TABLE] +NUM_LDO = 31 + + +[LDO_NAME1] +LDO_NAME=VTCXO1 +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_ldo_enable(MT6331_POWER_LDO_VTCXO1,KAL_TRUE); +[LDO_ENABLE_OFF1] +pmic_ldo_enable(MT6331_POWER_LDO_VTCXO1,KAL_FALSE); +[LDO_APPNAME1] +MT6331_POWER_LDO_VTCXO1 + +[LDO_NAME2] +LDO_NAME=VTCXO2 +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_ldo_enable(MT6331_POWER_LDO_VTCXO2,KAL_TRUE); +[LDO_ENABLE_OFF2] +pmic_ldo_enable(MT6331_POWER_LDO_VTCXO2,KAL_FALSE); +[LDO_APPNAME2] +MT6331_POWER_LDO_VTCXO2 + +[LDO_NAME3] +LDO_NAME=VAUD32 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_ldo_enable(MT6331_POWER_LDO_VAUD32,KAL_TRUE); +[LDO_ENABLE_OFF3] +pmic_ldo_enable(MT6331_POWER_LDO_VAUD32,KAL_FALSE); +[LDO_APPNAME3] +MT6331_POWER_LDO_VAUD32 + +[LDO_NAME4] +LDO_NAME=VAUXA32 +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_ldo_enable(MT6331_POWER_LDO_VAUXA32,KAL_TRUE); +[LDO_ENABLE_OFF4] +pmic_ldo_enable(MT6331_POWER_LDO_VAUXA32,KAL_FALSE); +[LDO_APPNAME4] +MT6331_POWER_LDO_VAUXA32 + +[LDO_NAME5] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_ldo_enable(MT6331_POWER_LDO_VCAMA,KAL_TRUE); +[LDO_ENABLE_OFF5] +pmic_ldo_enable(MT6331_POWER_LDO_VCAMA,KAL_FALSE); +[LDO_APPNAME5] +MT6331_POWER_LDO_VCAMA + +[LDO_NAME6] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_ldo_enable(MT6331_POWER_LDO_VMCH,KAL_TRUE); +[LDO_ENABLE_OFF6] +pmic_ldo_enable(MT6331_POWER_LDO_VMCH,KAL_FALSE); +[LDO_APPNAME6] +MT6331_POWER_LDO_VMCH + +[LDO_NAME7] +LDO_NAME=VEMC33 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_ldo_enable(MT6331_POWER_LDO_VEMC33,KAL_TRUE); +[LDO_ENABLE_OFF7] +pmic_ldo_enable(MT6331_POWER_LDO_VEMC33,KAL_FALSE); +[LDO_APPNAME7] +MT6331_POWER_LDO_VEMC33 + +[LDO_NAME8] +LDO_NAME=VIO28 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_ldo_enable(MT6331_POWER_LDO_VIO28,KAL_TRUE); +[LDO_ENABLE_OFF8] +pmic_ldo_enable(MT6331_POWER_LDO_VIO28,KAL_FALSE); +[LDO_APPNAME8] +MT6331_POWER_LDO_VIO28 + +[LDO_NAME9] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_ldo_enable(MT6331_POWER_LDO_VMC,KAL_TRUE); +[LDO_ENABLE_OFF9] +pmic_ldo_enable(MT6331_POWER_LDO_VMC,KAL_FALSE); +[LDO_APPNAME9] +MT6331_POWER_LDO_VMC + +[LDO_NAME10] +LDO_NAME=VCAM_AF +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_ldo_enable(MT6331_POWER_LDO_VCAM_AF,KAL_TRUE); +[LDO_ENABLE_OFF10] +pmic_ldo_enable(MT6331_POWER_LDO_VCAM_AF,KAL_FALSE); +[LDO_APPNAME10] +MT6331_POWER_LDO_VCAM_AF + +[LDO_NAME11] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_ldo_enable(MT6331_POWER_LDO_VGP1,KAL_TRUE); +[LDO_ENABLE_OFF11] +pmic_ldo_enable(MT6331_POWER_LDO_VGP1,KAL_FALSE); +[LDO_APPNAME11] +MT6331_POWER_LDO_VGP1 + +[LDO_NAME12] +LDO_NAME=VGP4 +[LDO_ENABLE_CONFIGURABLE12] +CONFIGURABLE = YES +[LDO_ENABLE_ON12] +pmic_ldo_enable(MT6331_POWER_LDO_VGP4,KAL_TRUE); +[LDO_ENABLE_OFF12] +pmic_ldo_enable(MT6331_POWER_LDO_VGP4,KAL_FALSE); +[LDO_APPNAME12] +MT6331_POWER_LDO_VGP4 + +[LDO_NAME13] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE13] +CONFIGURABLE = YES +[LDO_ENABLE_ON13] +pmic_ldo_enable(MT6331_POWER_LDO_VSIM1,KAL_TRUE); +[LDO_ENABLE_OFF13] +pmic_ldo_enable(MT6331_POWER_LDO_VSIM1,KAL_FALSE); +[LDO_APPNAME13] +MT6331_POWER_LDO_VSIM1 + +[LDO_NAME14] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE14] +CONFIGURABLE = YES +[LDO_ENABLE_ON14] +pmic_ldo_enable(MT6331_POWER_LDO_VSIM2,KAL_TRUE); +[LDO_ENABLE_OFF14] +pmic_ldo_enable(MT6331_POWER_LDO_VSIM2,KAL_FALSE); +[LDO_APPNAME14] +MT6331_POWER_LDO_VSIM2 + +[LDO_NAME15] +LDO_NAME=VFBB +[LDO_ENABLE_CONFIGURABLE15] +CONFIGURABLE = YES +[LDO_ENABLE_ON15] +pmic_ldo_enable(MT6331_POWER_LDO_VFBB,KAL_TRUE); +[LDO_ENABLE_OFF15] +pmic_ldo_enable(MT6331_POWER_LDO_VFBB,KAL_FALSE); +[LDO_APPNAME15] +MT6331_POWER_LDO_VFBB + +[LDO_NAME16] +LDO_NAME=VRTC +[LDO_ENABLE_CONFIGURABLE16] +CONFIGURABLE = YES +[LDO_ENABLE_ON16] +pmic_ldo_enable(MT6331_POWER_LDO_VRTC,KAL_TRUE); +[LDO_ENABLE_OFF16] +pmic_ldo_enable(MT6331_POWER_LDO_VRTC,KAL_FALSE); +[LDO_APPNAME16] +MT6331_POWER_LDO_VRTC + +[LDO_NAME17] +LDO_NAME=VMIPI +[LDO_ENABLE_CONFIGURABLE17] +CONFIGURABLE = YES +[LDO_ENABLE_ON17] +pmic_ldo_enable(MT6331_POWER_LDO_VMIPI,KAL_TRUE); +[LDO_ENABLE_OFF17] +pmic_ldo_enable(MT6331_POWER_LDO_VMIPI,KAL_FALSE); +[LDO_APPNAME17] +MT6331_POWER_LDO_VMIPI + +[LDO_NAME18] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE18] +CONFIGURABLE = YES +[LDO_ENABLE_ON18] +pmic_ldo_enable(MT6331_POWER_LDO_VIBR,KAL_TRUE); +[LDO_ENABLE_OFF18] +pmic_ldo_enable(MT6331_POWER_LDO_VIBR,KAL_FALSE); +[LDO_APPNAME18] +MT6331_POWER_LDO_VIBR + +[LDO_NAME19] +LDO_NAME=MT6331_VDIG18 +[LDO_ENABLE_CONFIGURABLE19] +CONFIGURABLE = YES +[LDO_ENABLE_ON19] +pmic_ldo_enable(MT6331_POWER_LDO_VDIG18,KAL_TRUE); +[LDO_ENABLE_OFF19] +pmic_ldo_enable(MT6331_POWER_LDO_VDIG18,KAL_FALSE); +[LDO_APPNAME19] +MT6331_POWER_LDO_VDIG18 + +[LDO_NAME20] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE20] +CONFIGURABLE = YES +[LDO_ENABLE_ON20] +pmic_ldo_enable(MT6331_POWER_LDO_VCAMD,KAL_TRUE); +[LDO_ENABLE_OFF20] +pmic_ldo_enable(MT6331_POWER_LDO_VCAMD,KAL_FALSE); +[LDO_APPNAME20] +MT6331_POWER_LDO_VCAMD + +[LDO_NAME21] +LDO_NAME=VUSB10 +[LDO_ENABLE_CONFIGURABLE21] +CONFIGURABLE = YES +[LDO_ENABLE_ON21] +pmic_ldo_enable(MT6331_POWER_LDO_VUSB10,KAL_TRUE); +[LDO_ENABLE_OFF21] +pmic_ldo_enable(MT6331_POWER_LDO_VUSB10,KAL_FALSE); +[LDO_APPNAME21] +MT6331_POWER_LDO_VUSB10 + +[LDO_NAME22] +LDO_NAME=VCAM_IO +[LDO_ENABLE_CONFIGURABLE22] +CONFIGURABLE = YES +[LDO_ENABLE_ON22] +pmic_ldo_enable(MT6331_POWER_LDO_VCAM_IO,KAL_TRUE); +[LDO_ENABLE_OFF22] +pmic_ldo_enable(MT6331_POWER_LDO_VCAM_IO,KAL_FALSE); +[LDO_APPNAME22] +MT6331_POWER_LDO_VCAM_IO + +[LDO_NAME23] +LDO_NAME=VSRAM_DVFS1 +[LDO_ENABLE_CONFIGURABLE23] +CONFIGURABLE = YES +[LDO_ENABLE_ON23] +pmic_ldo_enable(MT6331_POWER_LDO_VSRAM_DVFS1,KAL_TRUE); +[LDO_ENABLE_OFF23] +pmic_ldo_enable(MT6331_POWER_LDO_VSRAM_DVFS1,KAL_FALSE); +[LDO_APPNAME23] +MT6331_POWER_LDO_VSRAM_DVFS1 + +[LDO_NAME24] +LDO_NAME=VGP2 +[LDO_ENABLE_CONFIGURABLE24] +CONFIGURABLE = YES +[LDO_ENABLE_ON24] +pmic_ldo_enable(MT6331_POWER_LDO_VGP2,KAL_TRUE); +[LDO_ENABLE_OFF24] +pmic_ldo_enable(MT6331_POWER_LDO_VGP2,KAL_FALSE); +[LDO_APPNAME24] +MT6331_POWER_LDO_VGP2 + +[LDO_NAME25] +LDO_NAME=VGP3 +[LDO_ENABLE_CONFIGURABLE25] +CONFIGURABLE = YES +[LDO_ENABLE_ON25] +pmic_ldo_enable(MT6331_POWER_LDO_VGP3,KAL_TRUE); +[LDO_ENABLE_OFF25] +pmic_ldo_enable(MT6331_POWER_LDO_VGP3,KAL_FALSE); +[LDO_APPNAME25] +MT6331_POWER_LDO_VGP3 + +[LDO_NAME26] +LDO_NAME=VBIASN +[LDO_ENABLE_CONFIGURABLE26] +CONFIGURABLE = YES +[LDO_ENABLE_ON26] +pmic_ldo_enable(MT6331_POWER_LDO_VBIASN,KAL_TRUE); +[LDO_ENABLE_OFF26] +pmic_ldo_enable(MT6331_POWER_LDO_VBIASN,KAL_FALSE); +[LDO_APPNAME26] +MT6331_POWER_LDO_VBIASN + +[LDO_NAME27] +LDO_NAME=VBIF28 +[LDO_ENABLE_CONFIGURABLE27] +CONFIGURABLE = YES +[LDO_ENABLE_ON27] +pmic_ldo_enable(MT6332_POWER_LDO_VBIF28,KAL_TRUE); +[LDO_ENABLE_OFF27] +pmic_ldo_enable(MT6332_POWER_LDO_VBIF28,KAL_FALSE); +[LDO_APPNAME27] +MT6332_POWER_LDO_VBIF28 + +[LDO_NAME28] +LDO_NAME=VAUXB32 +[LDO_ENABLE_CONFIGURABLE28] +CONFIGURABLE = YES +[LDO_ENABLE_ON28] +pmic_ldo_enable(MT6332_POWER_LDO_VAUXB32,KAL_TRUE); +[LDO_ENABLE_OFF28] +pmic_ldo_enable(MT6332_POWER_LDO_VAUXB32,KAL_FALSE); +[LDO_APPNAME28] +MT6332_POWER_LDO_VAUXB32 + +[LDO_NAME29] +LDO_NAME=VUSB33 +[LDO_ENABLE_CONFIGURABLE29] +CONFIGURABLE = YES +[LDO_ENABLE_ON29] +pmic_ldo_enable(MT6332_POWER_LDO_VUSB33,KAL_TRUE); +[LDO_ENABLE_OFF29] +pmic_ldo_enable(MT6332_POWER_LDO_VUSB33,KAL_FALSE); +[LDO_APPNAME29] +MT6332_POWER_LDO_VUSB33 + +[LDO_NAME30] +LDO_NAME=MT6332_VDIG18 +[LDO_ENABLE_CONFIGURABLE30] +CONFIGURABLE = YES +[LDO_ENABLE_ON30] +pmic_ldo_enable(MT6332_POWER_LDO_VDIG18,KAL_TRUE); +[LDO_ENABLE_OFF30] +pmic_ldo_enable(MT6332_POWER_LDO_VDIG18,KAL_FALSE); +[LDO_APPNAME30] +MT6332_POWER_LDO_VDIG18 + +[LDO_NAME31] +LDO_NAME=VSRAM_DVFS2 +[LDO_ENABLE_CONFIGURABLE31] +CONFIGURABLE = YES +[LDO_ENABLE_ON31] +pmic_ldo_enable(MT6332_POWER_LDO_VSRAM_DVFS2,KAL_TRUE); +[LDO_ENABLE_OFF31] +pmic_ldo_enable(MT6332_POWER_LDO_VSRAM_DVFS2,KAL_FALSE); +[LDO_APPNAME31] +MT6332_POWER_LDO_VSRAM_DVFS2 + +[LDO_APPNAME_DEFAULT] +MT65XX_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6350PMUMP.cmp b/tools/dct/PMIC_MT6350PMUMP.cmp new file mode 100755 index 000000000..2229955f1 --- /dev/null +++ b/tools/dct/PMIC_MT6350PMUMP.cmp @@ -0,0 +1,218 @@ +[PMIC_TABLE] +NUM_LDO = 18 + + +[LDO_NAME1] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_set_register_value(PMIC_RG_VMC_EN,1); +[LDO_ENABLE_OFF1] +pmic_set_register_value(PMIC_RG_VMC_EN,0); +[LDO_APPNAME1] +"VMC" + +[LDO_NAME2] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_set_register_value(PMIC_RG_VMCH_EN,1); +[LDO_ENABLE_OFF2] +pmic_set_register_value(PMIC_RG_VMCH_EN,0); +[LDO_APPNAME2] +"VMCH" + +[LDO_NAME3] +LDO_NAME=VEMC_3V3 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_set_register_value(PMIC_RG_VEMC_3V3_EN,1); +[LDO_ENABLE_OFF3] +pmic_set_register_value(PMIC_RG_VEMC_3V3_EN,0); +[LDO_APPNAME3] +"VEMC_3V3" + +[LDO_NAME4] +LDO_NAME=VGP1 +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_set_register_value(PMIC_RG_VGP1_EN,1); +[LDO_ENABLE_OFF4] +pmic_set_register_value(PMIC_RG_VGP1_EN,0); +[LDO_APPNAME4] +"VGP1" + +[LDO_NAME5] +LDO_NAME=VGP2 +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_set_register_value(PMIC_RG_VGP2_EN,1); +[LDO_ENABLE_OFF5] +pmic_set_register_value(PMIC_RG_VGP2_EN,0); +[LDO_APPNAME5] +"VGP2" + +[LDO_NAME6] +LDO_NAME=VGP3 +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_set_register_value(PMIC_RG_VGP3_EN,1); +[LDO_ENABLE_OFF6] +pmic_set_register_value(PMIC_RG_VGP3_EN,0); +[LDO_APPNAME6] +"VGP3" + +[LDO_NAME7] +LDO_NAME=VCN_1V8 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_set_register_value(PMIC_RG_VCN_1V8_EN,1); +[LDO_ENABLE_OFF7] +pmic_set_register_value(PMIC_RG_VCN_1V8_EN,0); +[LDO_APPNAME7] +"VCN_1V8" + +[LDO_NAME8] +LDO_NAME=VSIM1 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_set_register_value(PMIC_RG_VSIM1_EN,1); +[LDO_ENABLE_OFF8] +pmic_set_register_value(PMIC_RG_VSIM1_EN,0); +[LDO_APPNAME8] +"VSIM1" + +[LDO_NAME9] +LDO_NAME=VSIM2 +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_set_register_value(PMIC_RG_VSIM2_EN,1); +[LDO_ENABLE_OFF9] +pmic_set_register_value(PMIC_RG_VSIM2_EN,0); +[LDO_APPNAME9] +"VSIM2" + +[LDO_NAME10] +LDO_NAME=VCAMAF +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_set_register_value(PMIC_RG_VCAM_AF_EN,1); +[LDO_ENABLE_OFF10] +pmic_set_register_value(PMIC_RG_VCAM_AF_EN,0); +[LDO_APPNAME10] +"VCAMAF" + +[LDO_NAME11] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_set_register_value(PMIC_RG_VIBR_EN,1); +[LDO_ENABLE_OFF11] +pmic_set_register_value(PMIC_RG_VIBR_EN,0); +[LDO_APPNAME11] +"VIBR" + +[LDO_NAME12] +LDO_NAME=VM +[LDO_ENABLE_CONFIGURABLE12] +CONFIGURABLE = YES +[LDO_ENABLE_ON12] +pmic_set_register_value(PMIC_RG_VM_EN,1); +[LDO_ENABLE_OFF12] +pmic_set_register_value(PMIC_RG_VM_EN,0); +[LDO_APPNAME12] +"VM" + +[LDO_NAME13] +LDO_NAME=VRF18 +[LDO_ENABLE_CONFIGURABLE13] +CONFIGURABLE = YES +[LDO_ENABLE_ON13] +pmic_set_register_value(PMIC_RG_VRF18_EN,1); +[LDO_ENABLE_OFF13] +pmic_set_register_value(PMIC_RG_VRF18_EN,0); +[LDO_APPNAME13] +"VRF18" + +[LDO_NAME14] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE14] +CONFIGURABLE = YES +[LDO_ENABLE_ON14] +pmic_set_register_value(PMIC_RG_VCAMD_EN,1); +[LDO_ENABLE_OFF14] +pmic_set_register_value(PMIC_RG_VCAMD_EN,0); +[LDO_APPNAME14] +"VCAMD" + +[LDO_NAME15] +LDO_NAME=VCANIO +[LDO_ENABLE_CONFIGURABLE15] +CONFIGURABLE = YES +[LDO_ENABLE_ON15] +pmic_set_register_value(PMIC_RG_VCAM_IO_EN,1); +[LDO_ENABLE_OFF15] +pmic_set_register_value(PMIC_RG_VCAM_IO_EN,0); +[LDO_APPNAME15] +"VCAMIO" + +[LDO_NAME16] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE16] +CONFIGURABLE = YES +[LDO_ENABLE_ON16] +pmic_set_register_value(PMIC_RG_VCAMA_EN,1); +[LDO_ENABLE_OFF16] +pmic_set_register_value(PMIC_RG_VCAMA_EN,0); +[LDO_APPNAME16] +"VCAMA" + +[LDO_NAME17] +LDO_NAME=VCN33 +[LDO_ENABLE_CONFIGURABLE17] +CONFIGURABLE = YES +[LDO_ENABLE_ON17] +pmic_set_register_value(PMIC_RG_VCN33_EN_WIFI,1); +[LDO_ENABLE_OFF17] +pmic_set_register_value(PMIC_RG_VCN33_EN_WIFI,0); +[LDO_APPNAME17] +"VCN33_WIFI" + +[LDO_NAME18] +LDO_NAME=VCN28 +[LDO_ENABLE_CONFIGURABLE18] +CONFIGURABLE = YES +[LDO_ENABLE_ON18] +pmic_set_register_value(PMIC_RG_VCN28_EN,1); +[LDO_ENABLE_OFF18] +pmic_set_register_value(PMIC_RG_VCN28_EN,0); +[LDO_APPNAME18] +"VCN28" + + +[LDO_APPNAME_DEFAULT] +"" + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_MT6397PMUMP.cmp b/tools/dct/PMIC_MT6397PMUMP.cmp new file mode 100755 index 000000000..c266108ac --- /dev/null +++ b/tools/dct/PMIC_MT6397PMUMP.cmp @@ -0,0 +1,137 @@ +[PMIC_TABLE] +NUM_LDO = 11 + +[LDO_NAME1] +LDO_NAME=VMC +[LDO_ENABLE_CONFIGURABLE1] +CONFIGURABLE = YES +[LDO_ENABLE_ON1] +pmic_ldo_enable(MT65XX_POWER_LDO_VMC,KAL_TRUE); +[LDO_ENABLE_OFF1] +pmic_ldo_enable(MT65XX_POWER_LDO_VMC,KAL_FALSE); +[LDO_APPNAME1] +MT65XX_POWER_LDO_VMC + +[LDO_NAME2] +LDO_NAME=VMCH +[LDO_ENABLE_CONFIGURABLE2] +CONFIGURABLE = YES +[LDO_ENABLE_ON2] +pmic_ldo_enable(MT65XX_POWER_LDO_VMCH,KAL_TRUE); +[LDO_ENABLE_OFF2] +pmic_ldo_enable(MT65XX_POWER_LDO_VMCH,KAL_FALSE); +[LDO_APPNAME2] +MT65XX_POWER_LDO_VMCH + +[LDO_NAME3] +LDO_NAME=VEMC_3V3 +[LDO_ENABLE_CONFIGURABLE3] +CONFIGURABLE = YES +[LDO_ENABLE_ON3] +pmic_ldo_enable(MT65XX_POWER_LDO_VEMC_3V3,KAL_TRUE); +[LDO_ENABLE_OFF3] +pmic_ldo_enable(MT65XX_POWER_LDO_VEMC_3V3,KAL_FALSE); +[LDO_APPNAME3] +MT65XX_POWER_LDO_VEMC_3V3 + +[LDO_NAME4] +LDO_NAME=VCAMD +[LDO_ENABLE_CONFIGURABLE4] +CONFIGURABLE = YES +[LDO_ENABLE_ON4] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMD,KAL_TRUE); +[LDO_ENABLE_OFF4] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMD,KAL_FALSE); +[LDO_APPNAME4] +MT65XX_POWER_LDO_VCAMD + +[LDO_NAME5] +LDO_NAME=VCAMIO +[LDO_ENABLE_CONFIGURABLE5] +CONFIGURABLE = YES +[LDO_ENABLE_ON5] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMIO,KAL_TRUE); +[LDO_ENABLE_OFF5] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMIO,KAL_FALSE); +[LDO_APPNAME5] +MT65XX_POWER_LDO_VCAMIO + +[LDO_NAME6] +LDO_NAME=VCAMAF +[LDO_ENABLE_CONFIGURABLE6] +CONFIGURABLE = YES +[LDO_ENABLE_ON6] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMAF,KAL_TRUE); +[LDO_ENABLE_OFF6] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMAF,KAL_FALSE); +[LDO_APPNAME6] +MT65XX_POWER_LDO_VCAMAF + +[LDO_NAME7] +LDO_NAME=VGP4 +[LDO_ENABLE_CONFIGURABLE7] +CONFIGURABLE = YES +[LDO_ENABLE_ON7] +pmic_ldo_enable(MT65XX_POWER_LDO_VGP4,KAL_TRUE); +[LDO_ENABLE_OFF7] +pmic_ldo_enable(MT65XX_POWER_LDO_VGP4,KAL_FALSE); +[LDO_APPNAME7] +MT65XX_POWER_LDO_VGP4 + +[LDO_NAME8] +LDO_NAME=VGP5 +[LDO_ENABLE_CONFIGURABLE8] +CONFIGURABLE = YES +[LDO_ENABLE_ON8] +pmic_ldo_enable(MT65XX_POWER_LDO_VGP5,KAL_TRUE); +[LDO_ENABLE_OFF8] +pmic_ldo_enable(MT65XX_POWER_LDO_VGP5,KAL_FALSE); +[LDO_APPNAME8] +MT65XX_POWER_LDO_VGP5 + +[LDO_NAME9] +LDO_NAME=VGP6 +[LDO_ENABLE_CONFIGURABLE9] +CONFIGURABLE = YES +[LDO_ENABLE_ON9] +pmic_ldo_enable(MT65XX_POWER_LDO_VGP6,KAL_TRUE); +[LDO_ENABLE_OFF9] +pmic_ldo_enable(MT65XX_POWER_LDO_VGP6,KAL_FALSE); +[LDO_APPNAME9] +MT65XX_POWER_LDO_VGP6 + +[LDO_NAME10] +LDO_NAME=VIBR +[LDO_ENABLE_CONFIGURABLE10] +CONFIGURABLE = YES +[LDO_ENABLE_ON10] +pmic_ldo_enable(MT65XX_POWER_LDO_VIBR,KAL_TRUE); +[LDO_ENABLE_OFF10] +pmic_ldo_enable(MT65XX_POWER_LDO_VIBR,KAL_FALSE); +[LDO_APPNAME10] +MT65XX_POWER_LDO_VIBR + +[LDO_NAME11] +LDO_NAME=VCAMA +[LDO_ENABLE_CONFIGURABLE11] +CONFIGURABLE = YES +[LDO_ENABLE_ON11] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMA,KAL_TRUE); +[LDO_ENABLE_OFF11] +pmic_ldo_enable(MT65XX_POWER_LDO_VCAMA,KAL_FALSE); +[LDO_APPNAME11] +MT65XX_POWER_LDO_VCAMA + +[LDO_APPNAME_DEFAULT] +MT65XX_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/PMIC_NCPMU.cmp b/tools/dct/PMIC_NCPMU.cmp new file mode 100755 index 000000000..7f27fa8b8 --- /dev/null +++ b/tools/dct/PMIC_NCPMU.cmp @@ -0,0 +1,17 @@ +[PMIC_TABLE] +NUM_LDO = 0 + + +[LDO_APPNAME_DEFAULT] +MT6323_POWER_LDO_DEFAULT + +[pmic_drv.h_HEADER] + +[pmic_drv.h_TAILER] + +[pmic_drv.c_HEADER] +#include +#include +#include + +[pmic_drv.c_TAILER] diff --git a/tools/dct/UEM.cmp b/tools/dct/UEM.cmp new file mode 100755 index 000000000..2859d7273 --- /dev/null +++ b/tools/dct/UEM.cmp @@ -0,0 +1,13 @@ +[UEM] +MAX_NETNAME_TEXT = 16 + +[uem_drv.c_HEADER] +#ifdef __CUST_NEW__ +#include "kal_release.h" +#include "gpio_drv.h" +#include "custom_em.h" +#include "custom_equipment.h" + + +[uem_drv.c_TAILER] +#endif /* __CUST_NEW__ */ diff --git a/tools/dct/Wizard/AST.cmp b/tools/dct/Wizard/AST.cmp new file mode 100755 index 000000000..39f438ebd --- /dev/null +++ b/tools/dct/Wizard/AST.cmp @@ -0,0 +1,29 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = AST +[INFO_0_CONTENT] +Enable +Disable + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = Enable +[GPIO_NAME0] +GPIO_AST_RST_PIN +GPIO_AST_CS_PIN +GPIO_AST_CLK32K_PIN +GPIO_AST_WAKEUP_PIN +GPIO_AST_INTR_PIN +GPIO_AST_WAKEUP_INTR_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = Disable +[GPIO_NAME1] + +[EINT_NAME1] diff --git a/tools/dct/Wizard/Bluetooth.cmp b/tools/dct/Wizard/Bluetooth.cmp new file mode 100755 index 000000000..70319be26 --- /dev/null +++ b/tools/dct/Wizard/Bluetooth.cmp @@ -0,0 +1,44 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = BT Chip Type +[INFO_0_CONTENT] +MT6611/MT6612/MT6616 +MT6620 + + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = MT6611/MT6612/MT6616 +[GPIO_NAME0] +GPIO_BT_POWREN_PIN +GPIO_BT_RESET_PIN +GPIO_BT_EINT_PIN +GPIO_BT_CLK_PIN +GPIO_PCM_DAICLK_PIN +GPIO_PCM_DAIPCMOUT_PIN +GPIO_PCM_DAIPCMIN_PIN +GPIO_PCM_DAISYNC_PIN +[EINT_NAME0] +BT + +[TYPE1] +INFO_0 = MT6620 +[GPIO_NAME1] +GPIO_COMBO_6620_LDO_EN_PIN +GPIO_COMBO_PMU_EN_PIN +GPIO_COMBO_RST_PIN +GPIO_COMBO_RTCCLK_PIN +GPIO_COMBO_BGF_EINT_PIN +GPIO_COMBO_ALL_EINT_PIN +GPIO_PCM_DAICLK_PIN +GPIO_PCM_DAIPCMOUT_PIN +GPIO_PCM_DAIPCMIN_PIN +GPIO_PCM_DAISYNC_PIN +[EINT_NAME1] +COMBO_BGF +COMBO_ALL diff --git a/tools/dct/Wizard/CMMB.cmp b/tools/dct/Wizard/CMMB.cmp new file mode 100755 index 000000000..2f795be68 --- /dev/null +++ b/tools/dct/Wizard/CMMB.cmp @@ -0,0 +1,31 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = Power Source Select +[INFO_0_CONTENT] +Ext_LDO +Int_LDO + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + + +[TYPE0] +INFO_0 = Ext_LDO +[GPIO_NAME0] +GPIO_CMMB_EINT_PIN +GPIO_CMMB_LDO_EN_PIN +GPIO_CMMB_RST_PIN +[EINT_NAME0] +CMMB + +[TYPE1] +INFO_0 = Int_LDO +[GPIO_NAME1] +GPIO_CMMB_EINT_PIN +GPIO_CMMB_RST_PIN +[EINT_NAME1] +CMMB diff --git a/tools/dct/Wizard/Camera.cmp b/tools/dct/Wizard/Camera.cmp new file mode 100755 index 000000000..54b16d123 --- /dev/null +++ b/tools/dct/Wizard/Camera.cmp @@ -0,0 +1,127 @@ +[MODULE_INFO] +INFO_NUM = 3 + +[INFO_0_NAME] +INFO_NAME = Camera Power Source +[INFO_0_CONTENT] +Ext_LDO +Int_LDO + +[INFO_1_NAME] +INFO_NAME = AF Power Source +[INFO_1_CONTENT] +Ext_LDO +Int_LDO + +[INFO_2_NAME] +INFO_NAME = Flashlight Enable/Disable +[INFO_2_CONTENT] +Enable +Disable + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = Ext_LDO +INFO_1 = Int_LDO +INFO_2 = Disable +[GPIO_NAME0] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_AF_EN_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = Ext_LDO +INFO_1 = Int_LDO +INFO_2 = Enable +[GPIO_NAME1] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_FLASH_EN_PIN +GPIO_CAMERA_FLASH_MODE_PIN +[EINT_NAME1] + +[TYPE2] +INFO_0 = Ext_LDO +INFO_1 = Ext_LDO +INFO_2 = Disable +[GPIO_NAME2] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +[EINT_NAME2] + +[TYPE3] +INFO_0 = Int_LDO +INFO_1 = Ext_LDO +INFO_2 = Disable +[GPIO_NAME3] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_LDO_EN_PIN +[EINT_NAME3] + +[TYPE4] +INFO_0 = Int_LDO +INFO_1 = Int_LDO +INFO_2 = Disable +[GPIO_NAME4] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_AF_EN_PIN +GPIO_CAMERA_LDO_EN_PIN +[EINT_NAME4] + +[TYPE5] +INFO_0 = Int_LDO +INFO_1 = Int_LDO +INFO_2 = Enable +[GPIO_NAME5] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_AF_EN_PIN +GPIO_CAMERA_LDO_EN_PIN +GPIO_CAMERA_FLASH_EN_PIN +GPIO_CAMERA_FLASH_MODE_PIN +[EINT_NAME5] + +[TYPE6] +INFO_0 = Int_LDO +INFO_1 = Ext_LDO +INFO_2 = Enable +[GPIO_NAME6] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_LDO_EN_PIN +GPIO_CAMERA_FLASH_EN_PIN +GPIO_CAMERA_FLASH_MODE_PIN +[EINT_NAME6] + +[TYPE7] +INFO_0 = Ext_LDO +INFO_1 = Ext_LDO +INFO_2 = Enable +[GPIO_NAME7] +GPIO_CAMERA_CMRST_PIN +GPIO_CAMERA_CMPDN_PIN +GPIO_CAMERA_CMRST1_PIN +GPIO_CAMERA_CMPDN1_PIN +GPIO_CAMERA_FLASH_EN_PIN +GPIO_CAMERA_FLASH_MODE_PIN +[EINT_NAME7] \ No newline at end of file diff --git a/tools/dct/Wizard/FM.cmp b/tools/dct/Wizard/FM.cmp new file mode 100755 index 000000000..00a234196 --- /dev/null +++ b/tools/dct/Wizard/FM.cmp @@ -0,0 +1,29 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = FM Chip Select +[INFO_0_CONTENT] +AR1000 +MT6616 + +[INFO_0_PROPERTY] +MULTIPLE_CHECK = NO + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = AR1000 +[GPIO_NAME0] +GPIO_FM_CLK_PIN +[EINT_NAME0] + + +[TYPE1] +INFO_0 = MT6616 +[GPIO_NAME1] +GPIO_FM_RDS_PIN +[EINT_NAME1] +FM_RDS \ No newline at end of file diff --git a/tools/dct/Wizard/GPS.cmp b/tools/dct/Wizard/GPS.cmp new file mode 100755 index 000000000..c222bd2a5 --- /dev/null +++ b/tools/dct/Wizard/GPS.cmp @@ -0,0 +1,42 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = GPS Chip Type +[INFO_0_CONTENT] +MT3326 +MT3320 + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = MT3326 +[GPIO_NAME0] +GPIO_GPS_PWREN_PIN +GPIO_GPS_SYNC_PIN +GPIO_GPS_CLK_PIN +GPIO_GPS_RST_PIN +[EINT_NAME0] +GPS + +[TYPE0] +INFO_0 = MT3320 +[GPIO_NAME0] +GPIO_GPS_SYNC_PIN +GPIO_COMBO_6620_LDO_EN_PIN +GPIO_COMBO_PMU_EN_PIN +GPIO_COMBO_RST_PIN +GPIO_COMBO_RTCCLK_PIN +GPIO_COMBO_BGF_EINT_PIN +GPIO_COMBO_ALL_EINT_PIN +GPIO_PCM_DAICLK_PIN +GPIO_PCM_DAIPCMOUT_PIN +GPIO_PCM_DAIPCMIN_PIN +GPIO_PCM_DAISYNC_PIN +[EINT_NAME0] +COMBO_BGF +COMBO_ALL + diff --git a/tools/dct/Wizard/HW_Module.cmp b/tools/dct/Wizard/HW_Module.cmp new file mode 100755 index 000000000..4d65ab792 --- /dev/null +++ b/tools/dct/Wizard/HW_Module.cmp @@ -0,0 +1,22 @@ +[Module_Name] +Bluetooth +GPS +WIFI +CMMB +FM +mATV +Camera +LCD +Sensors +Touch Panel +Keypad +Jogball and OFN +MISC +USB +AST +I2C +I2S +PWM +UART +SPI + diff --git a/tools/dct/Wizard/I2C.cmp b/tools/dct/Wizard/I2C.cmp new file mode 100755 index 000000000..6b3a00fd2 --- /dev/null +++ b/tools/dct/Wizard/I2C.cmp @@ -0,0 +1,38 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = I2C +[INFO_0_CONTENT] +I2C0 +I2C1 +I2C3 +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = I2C0 +[GPIO_NAME0] +GPIO_I2C0_SCA_PIN +GPIO_I2C0_SDA_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = I2C1 +[GPIO_NAME1] +GPIO_I2C1_SCA_PIN +GPIO_I2C1_SDA_PIN +[EINT_NAME1] + +[TYPE2] +INFO_0 = I2C2 +[GPIO_NAME2] +GPIO_I2C2_SCA_PIN +GPIO_I2C2_SDA_PIN +[EINT_NAME2] + diff --git a/tools/dct/Wizard/I2S.cmp b/tools/dct/Wizard/I2S.cmp new file mode 100755 index 000000000..00aa266a1 --- /dev/null +++ b/tools/dct/Wizard/I2S.cmp @@ -0,0 +1,31 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = I2S +[INFO_0_CONTENT] +I2S0 +I2S1 +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = I2S0 +[GPIO_NAME0] +GPIO_I2S0_CK_PIN +GPIO_I2S0_DAT_PIN +GPIO_I2S0_WS_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = I2S1 +[GPIO_NAME1] +GPIO_I2S1_CK_PIN +GPIO_I2S1_DAT_PIN +GPIO_I2S1_WS_PIN +[EINT_NAME1] diff --git a/tools/dct/Wizard/Jogball_and_OFN.cmp b/tools/dct/Wizard/Jogball_and_OFN.cmp new file mode 100755 index 000000000..66a26f898 --- /dev/null +++ b/tools/dct/Wizard/Jogball_and_OFN.cmp @@ -0,0 +1,83 @@ +[MODULE_INFO] +INFO_NUM = 2 + +[INFO_0_NAME] +INFO_NAME = Jogbal Enable/Disable +[INFO_0_CONTENT] +Enable +Disable + +[INFO_1_NAME] +INFO_NAME = OFN Enable/Disable +[INFO_1_CONTENT] +Enable +Disable + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = Enable +INFO_1 = Enable +[GPIO_NAME0] +GPIO_JBD_INPUT_UP_PIN +GPIO_JBD_INPUT_LEFT_PIN +GPIO_JBD_INPUT_RIGHT_PIN +GPIO_JBD_INPUT_DOWN_PIN +GPIO_HALL_1_PIN +GPIO_HALL_2_PIN +GPIO_HALL_3_PIN +GPIO_HALL_4_PIN +GPIO_OFN_EINT_PIN +GPIO_OFN_DWN_PIN +GPIO_OFN_RST_PIN +[EINT_NAME0] +HALL_1 +HALL_2 +HALL_3 +HALL_4 +OFN + +[TYPE1] +INFO_0 = Enabel +INFO_1 = Disable +[GPIO_NAME1] +GPIO_JBD_INPUT_UP_PIN +GPIO_JBD_INPUT_LEFT_PIN +GPIO_JBD_INPUT_RIGHT_PIN +GPIO_JBD_INPUT_DOWN_PIN +GPIO_HALL_1_PIN +GPIO_HALL_2_PIN +GPIO_HALL_3_PIN +GPIO_HALL_4_PIN +[EINT_NAME1] +HALL_1 +HALL_2 +HALL_3 +HALL_4 + +[TYPE2] +INFO_0 = Disable +INFO_1 = Enable +[GPIO_NAME2] +GPIO_OFN_EINT_PIN +GPIO_OFN_DWN_PIN +GPIO_OFN_RST_PIN +[EINT_NAME2] +OFN + +[TYPE3] +INFO_0 = Disable +INFO_1 = Disable +[GPIO_NAME3] + +[EINT_NAME3] + + + + + + + + diff --git a/tools/dct/Wizard/JogballandOFN_old.cmp b/tools/dct/Wizard/JogballandOFN_old.cmp new file mode 100755 index 000000000..66a26f898 --- /dev/null +++ b/tools/dct/Wizard/JogballandOFN_old.cmp @@ -0,0 +1,83 @@ +[MODULE_INFO] +INFO_NUM = 2 + +[INFO_0_NAME] +INFO_NAME = Jogbal Enable/Disable +[INFO_0_CONTENT] +Enable +Disable + +[INFO_1_NAME] +INFO_NAME = OFN Enable/Disable +[INFO_1_CONTENT] +Enable +Disable + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = Enable +INFO_1 = Enable +[GPIO_NAME0] +GPIO_JBD_INPUT_UP_PIN +GPIO_JBD_INPUT_LEFT_PIN +GPIO_JBD_INPUT_RIGHT_PIN +GPIO_JBD_INPUT_DOWN_PIN +GPIO_HALL_1_PIN +GPIO_HALL_2_PIN +GPIO_HALL_3_PIN +GPIO_HALL_4_PIN +GPIO_OFN_EINT_PIN +GPIO_OFN_DWN_PIN +GPIO_OFN_RST_PIN +[EINT_NAME0] +HALL_1 +HALL_2 +HALL_3 +HALL_4 +OFN + +[TYPE1] +INFO_0 = Enabel +INFO_1 = Disable +[GPIO_NAME1] +GPIO_JBD_INPUT_UP_PIN +GPIO_JBD_INPUT_LEFT_PIN +GPIO_JBD_INPUT_RIGHT_PIN +GPIO_JBD_INPUT_DOWN_PIN +GPIO_HALL_1_PIN +GPIO_HALL_2_PIN +GPIO_HALL_3_PIN +GPIO_HALL_4_PIN +[EINT_NAME1] +HALL_1 +HALL_2 +HALL_3 +HALL_4 + +[TYPE2] +INFO_0 = Disable +INFO_1 = Enable +[GPIO_NAME2] +GPIO_OFN_EINT_PIN +GPIO_OFN_DWN_PIN +GPIO_OFN_RST_PIN +[EINT_NAME2] +OFN + +[TYPE3] +INFO_0 = Disable +INFO_1 = Disable +[GPIO_NAME3] + +[EINT_NAME3] + + + + + + + + diff --git a/tools/dct/Wizard/Keypad.cmp b/tools/dct/Wizard/Keypad.cmp new file mode 100755 index 000000000..aa59e5094 --- /dev/null +++ b/tools/dct/Wizard/Keypad.cmp @@ -0,0 +1,121 @@ +[MODULE_INFO] +INFO_NUM = 2 + +[INFO_0_NAME] +INFO_NAME = Power Key Use EINT +[INFO_0_CONTENT] +Yes +No + +[INFO_0_PROPERTY] +MULTIPLE_CHECK = NO + +[INFO_1_NAME] +INFO_NAME = QWERTYSLIDE Keypad +[INFO_1_CONTENT] +Yes +No + +[INFO_1_PROPERTY] +MULTIPLE_CHECK = NO + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = Yes +INFO_1 = Yes +[GPIO_NAME0] +GPIO_QWERTYSLIDE_EINT_PIN +GPIO_PWR_BUTTON_PIN +GPIO_KPD_KCOL0_PIN +GPIO_KPD_KCOL1_PIN +GPIO_KPD_KCOL2_PIN +GPIO_KPD_KCOL3_PIN +GPIO_KPD_KCOL4_PIN +GPIO_KPD_KCOL5_PIN +GPIO_KPD_KCOL6_PIN +GPIO_KPD_KCOL7_PIN +GPIO_KPD_KROW0_PIN +GPIO_KPD_KROW1_PIN +GPIO_KPD_KROW2_PIN +GPIO_KPD_KROW3_PIN +GPIO_KPD_KROW4_PIN +GPIO_KPD_KROW5_PIN +GPIO_KPD_KROW6_PIN +GPIO_KPD_KROW7_PIN +[EINT_NAME0] +KPD_PWRKEY +KPD_SLIDE + +[TYPE1] +INFO_0 = Yes +INFO_1 = No +[GPIO_NAME1] +GPIO_PWR_BUTTON_PIN +GPIO_KPD_KCOL0_PIN +GPIO_KPD_KCOL1_PIN +GPIO_KPD_KCOL2_PIN +GPIO_KPD_KCOL3_PIN +GPIO_KPD_KCOL4_PIN +GPIO_KPD_KCOL5_PIN +GPIO_KPD_KCOL6_PIN +GPIO_KPD_KCOL7_PIN +GPIO_KPD_KROW0_PIN +GPIO_KPD_KROW1_PIN +GPIO_KPD_KROW2_PIN +GPIO_KPD_KROW3_PIN +GPIO_KPD_KROW4_PIN +GPIO_KPD_KROW5_PIN +GPIO_KPD_KROW6_PIN +GPIO_KPD_KROW7_PIN +[EINT_NAME1] +KPD_PWRKEY + +[TYPE2] +INFO_0 = No +INFO_1 = Yes +[GPIO_NAME2] +GPIO_QWERTYSLIDE_EINT_PIN +GPIO_KPD_KCOL0_PIN +GPIO_KPD_KCOL1_PIN +GPIO_KPD_KCOL2_PIN +GPIO_KPD_KCOL3_PIN +GPIO_KPD_KCOL4_PIN +GPIO_KPD_KCOL5_PIN +GPIO_KPD_KCOL6_PIN +GPIO_KPD_KCOL7_PIN +GPIO_KPD_KROW0_PIN +GPIO_KPD_KROW1_PIN +GPIO_KPD_KROW2_PIN +GPIO_KPD_KROW3_PIN +GPIO_KPD_KROW4_PIN +GPIO_KPD_KROW5_PIN +GPIO_KPD_KROW6_PIN +GPIO_KPD_KROW7_PIN +[EINT_NAME2] +KPD_SLIDE + +[TYPE3] +INFO_0 = No +INFO_1 = No +[GPIO_NAME2] +GPIO_KPD_KCOL0_PIN +GPIO_KPD_KCOL1_PIN +GPIO_KPD_KCOL2_PIN +GPIO_KPD_KCOL3_PIN +GPIO_KPD_KCOL4_PIN +GPIO_KPD_KCOL5_PIN +GPIO_KPD_KCOL6_PIN +GPIO_KPD_KCOL7_PIN +GPIO_KPD_KROW0_PIN +GPIO_KPD_KROW1_PIN +GPIO_KPD_KROW2_PIN +GPIO_KPD_KROW3_PIN +GPIO_KPD_KROW4_PIN +GPIO_KPD_KROW5_PIN +GPIO_KPD_KROW6_PIN +GPIO_KPD_KROW7_PIN +[EINT_NAME2] \ No newline at end of file diff --git a/tools/dct/Wizard/LCD.cmp b/tools/dct/Wizard/LCD.cmp new file mode 100755 index 000000000..3e16e4dc3 --- /dev/null +++ b/tools/dct/Wizard/LCD.cmp @@ -0,0 +1,32 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = LCD Type +[INFO_0_CONTENT] +MCU +RGB +MIPI + +[INFO_0_PROPERTY] +MULTIPLE_CHECK = NO + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = RGB +[GPIO_NAME0] +GPIO_DISP_LSCK_PIN +GPIO_DISP_LSA0_PIN +GPIO_DISP_LSDA_PIN +GPIO_DISP_LSCE_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = MCU +[GPIO_NAME1] +[EINT_NAME1] + + diff --git a/tools/dct/Wizard/MISC.cmp b/tools/dct/Wizard/MISC.cmp new file mode 100755 index 000000000..64e88863f --- /dev/null +++ b/tools/dct/Wizard/MISC.cmp @@ -0,0 +1,30 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = Please Select Function To Set +[INFO_0_CONTENT] +PMIC +External Audio AMP + +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = PMIC +[GPIO_NAME0] +GPIO_PMIC_EINT_PIN +[EINT_NAME0] +MT6326_PMIC + +[TYPE1] +INFO_0 = External Audio AMP +[GPIO_NAME1] +GPIO_SPEAKER_EN_PIN +[EINT_NAME1] + + diff --git a/tools/dct/Wizard/MotionSensor.cmp b/tools/dct/Wizard/MotionSensor.cmp new file mode 100755 index 000000000..38ba47008 --- /dev/null +++ b/tools/dct/Wizard/MotionSensor.cmp @@ -0,0 +1,156 @@ +[MODULE_INFO] +INFO_NUM = 2 + +[INFO_0_NAME] +INFO_NAME = Motion Chip Type +[INFO_0_CONTENT] +6601 +6611 + +[INFO_1_NAME] +INFO_NAME = Motion 26MHz_Clock_Source +[INFO_1_CONTENT] +Co_Clock with RF:AD6548 +BT Soc:MT6236 + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 +PROTECT_0 = gpio_bt_power_pin 0 0 1 0 1 0 0 +PROTECT_1 = gpio_bt_32k_pin 0 0 1 0 1 0 0 +PROTECT_2 = gpio_bt_utxd3_pin 0 0 1 0 1 0 0 +PROTECT_3 = gpio_bt_urxd3_pin 1 1 0 1 1 0 0 +PROTECT_4 = gpio_bt_pcmclk_pin 0 0 1 0 1 0 0 +PROTECT_5 = gpio_bt_pcmsync_pin 0 0 1 0 1 0 0 +PROTECT_6 = gpio_bt_pcmin_pin 1 0 0 1 1 0 0 +PROTECT_7 = gpio_bt_pcmout_pin 0 0 1 0 1 0 0 +PROTECT_8 = gpio_bt_reset_pin 0 0 1 0 1 0 0 +PROTECT_9 = gpio_bt_co_clock_pin 1 0 1 0 1 0 0 + +;non co-clock +;6601 ,6611 only support non co-clock +[TYPE0] +INFO_0 = 6601 +INFO_1 = Dedicate Clock source +[GPIO_NAME0] +gpio_bt_power_pin +gpio_bt_32k_pin +gpio_bt_utxd3_pin +gpio_bt_urxd3_pin +gpio_bt_pcmclk_pin +gpio_bt_pcmsync_pin +gpio_bt_pcmin_pin +gpio_bt_pcmout_pin +gpio_bt_reset_pin +[EINT_NAME0] +BT_CO_CLOCK_EINT_NO + +;6612 6616 co-clock with MT6253 MT6268 +[TYPE1] +INFO_0 = 6611 +INFO_1 = Dedicate Clock source +[GPIO_NAME1] +gpio_bt_power_pin +gpio_bt_32k_pin +gpio_bt_utxd3_pin +gpio_bt_urxd3_pin +gpio_bt_pcmclk_pin +gpio_bt_pcmsync_pin +gpio_bt_pcmin_pin +gpio_bt_pcmout_pin +gpio_bt_reset_pin +[EINT_NAME1] +BT_CO_CLOCK_EINT_NO + +;6612 6616 +[TYPE2] +INFO_0 = 6612 +INFO_1 = MT6253,MT6268 +[GPIO_NAME2] +GPIO_BT_POWREN_PIN +GPIO_BT_RESET_PIN +GPIO_BT_EINT_PIN +GPIO_BT_CLK_PIN +;gpio_bt_power_pin +;gpio_bt_32k_pin +;gpio_bt_utxd3_pin +;gpio_bt_urxd3_pin +;gpio_bt_pcmclk_pin +;gpio_bt_pcmsync_pin +;gpio_bt_pcmin_pin +;gpio_bt_pcmout_pin +;gpio_bt_reset_pin +;gpio_bt_co_clock_pin +[EINT_NAME2] +BT_CO_CLOCK_EINT_NO + +;6612 6616 +[TYPE3] +INFO_0 = 6616 +INFO_1 = MT6253,MT6268 +[GPIO_NAME3] +GPIO_BT_POWREN_PIN +GPIO_BT_RESET_PIN +GPIO_BT_EINT_PIN +GPIO_BT_CLK_PIN +;gpio_bt_power_pin +;gpio_bt_32k_pin +;gpio_bt_utxd3_pin +;gpio_bt_urxd3_pin +;gpio_bt_pcmclk_pin +;gpio_bt_pcmsync_pin +;gpio_bt_pcmin_pin +;gpio_bt_pcmout_pin +;gpio_bt_reset_pin +;gpio_bt_co_clock_pin +[EINT_NAME3] +BT_CO_CLOCK_EINT_NO + +[TYPE4] +INFO_0 = 6612 +INFO_1 = Co_Clock with RF:AD6548 +[GPIO_NAME4] +gpio_bt_power_pin +gpio_bt_32k_pin +gpio_bt_utxd3_pin +gpio_bt_urxd3_pin +gpio_bt_pcmclk_pin +gpio_bt_pcmsync_pin +gpio_bt_pcmin_pin +gpio_bt_pcmout_pin +gpio_bt_reset_pin +gpio_bt_co_clock_pin +[EINT_NAME4] +BT_CO_CLOCK_EINT_NO + +[TYPE5] +INFO_0 = 6616 +INFO_1 = Co_Clock with RF:AD6548 +[GPIO_NAME5] +gpio_bt_power_pin +gpio_bt_32k_pin +gpio_bt_utxd3_pin +gpio_bt_urxd3_pin +gpio_bt_pcmclk_pin +gpio_bt_pcmsync_pin +gpio_bt_pcmin_pin +gpio_bt_pcmout_pin +gpio_bt_reset_pin +gpio_bt_co_clock_pin +[EINT_NAME5] +BT_CO_CLOCK_EINT_NO + +;MT6236 +;no gpio and eint pin setting +[TYPE6] +INFO_0 = MT6236 +INFO_1 = Dedicate Clock source +[GPIO_NAME6] +;no gpio and eint pin setting +[EINT_NAME6] +BT_CO_CLOCK_EINT_NO + + + + + diff --git a/tools/dct/Wizard/PWM.cmp b/tools/dct/Wizard/PWM.cmp new file mode 100755 index 000000000..0f6ef6bfb --- /dev/null +++ b/tools/dct/Wizard/PWM.cmp @@ -0,0 +1,62 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = PWM +[INFO_0_CONTENT] +PWM1 +PWM2 +PWM3 +PWM4 +PWM5 +PWM6 +PWM7 +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = PWM1 +[GPIO_NAME0] +GPIO_PWM_1_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = PWM2 +[GPIO_NAME1] +GPIO_PWM_2_PIN +[EINT_NAME1] + +[TYPE2] +INFO_0 = PWM3 +[GPIO_NAME2] +GPIO_PWM_3_PIN +[EINT_NAME2] + +[TYPE3] +INFO_0 = PWM4 +[GPIO_NAME3] +GPIO_PWM_4_PIN +[EINT_NAME3] + +[TYPE4] +INFO_0 = PWM5 +[GPIO_NAME4] +GPIO_PWM_5_PIN +[EINT_NAME4] + +[TYPE5] +INFO_0 = PWM6 +[GPIO_NAME5] +GPIO_PWM_6_PIN +[EINT_NAME5] + +[TYPE6] +INFO_0 = PWM7 +[GPIO_NAME6] +GPIO_PWM_7_PIN +[EINT_NAME6] diff --git a/tools/dct/Wizard/SPI.cmp b/tools/dct/Wizard/SPI.cmp new file mode 100755 index 000000000..4d28318c1 --- /dev/null +++ b/tools/dct/Wizard/SPI.cmp @@ -0,0 +1,30 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = SPI +[INFO_0_CONTENT] +Enable +Disable +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = Enable +[GPIO_NAME0] +GPIO_SPI_CS_PIN +GPIO_SPI_SCK_PIN +GPIO_SPI_MISO_PIN +GPIO_SPI_MOSI_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = Disable +[GPIO_NAME1] + +[EINT_NAME1] diff --git a/tools/dct/Wizard/Sensors.cmp b/tools/dct/Wizard/Sensors.cmp new file mode 100755 index 000000000..1612d7d5e --- /dev/null +++ b/tools/dct/Wizard/Sensors.cmp @@ -0,0 +1,66 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = Please Select Sensors EINT Pin +[INFO_0_CONTENT] +ALS +GSE_1 +GSE_2 +MSE +GYRO +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = ALS +[GPIO_NAME0] +GPIO_ALS_EINT_PIN +[EINT_NAME0] +ALS + +[TYPE1] +INFO_0 = GSE_1 +[GPIO_NAME1] +GPIO_GSE_1_EINT_PIN +[EINT_NAME1] +GSE_1 + +[TYPE2] +INFO_0 = GSE_2 +[GPIO_NAME2] +GPIO_GSE_2_EINT_PIN +[EINT_NAME2] +GSE_1 + +[TYPE3] +INFO_0 = MSE +[GPIO_NAME3] +GPIO_MSE_EINT_PIN +[EINT_NAME3] +MSE + +[TYPE4] +INFO_0 = GYRO +[GPIO_NAME4] +GPIO_GYRO_EINT_PIN +[EINT_NAME4] +GYRO + +[TYPE5] +INFO_0 = PWM6 +[GPIO_NAME5] +GPIO_PWM_6_PIN +[EINT_NAME5] + +[TYPE6] +INFO_0 = PWM7 +[GPIO_NAME6] +GPIO_PWM_7_PIN +[EINT_NAME6] + diff --git a/tools/dct/Wizard/TouchPanel.cmp b/tools/dct/Wizard/TouchPanel.cmp new file mode 100755 index 000000000..6d96f83d0 --- /dev/null +++ b/tools/dct/Wizard/TouchPanel.cmp @@ -0,0 +1,33 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = Touch Panel Type +[INFO_0_CONTENT] +R-touch +C-touch + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = R-touch +[GPIO_NAME0] + +[EINT_NAME0] + + +[TYPE1] +INFO_0 = C-touch +[GPIO_NAME1] +GPIO_CTP_EINT_PIN +GPIO_CTP_EN_PIN +GPIO_CTP_RST_PIN +[EINT_NAME1] +TOUCH_PANEL + + + + + diff --git a/tools/dct/Wizard/UART.cmp b/tools/dct/Wizard/UART.cmp new file mode 100755 index 000000000..e8e2154b0 --- /dev/null +++ b/tools/dct/Wizard/UART.cmp @@ -0,0 +1,53 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = UART +[INFO_0_CONTENT] +UART1 +UART2 +UART3 +UART4 +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = UART1 +[GPIO_NAME0] +GPIO_UART_URXD1_PIN +GPIO_UART_UTXD1_PIN +GPIO_UART_UCTS1_PIN +GPIO_UART_URTS1_PIN +[EINT_NAME0] + +[TYPE1] +INFO_0 = UART2 +[GPIO_NAME1] +GPIO_UART_URXD2_PIN +GPIO_UART_UTXD2_PIN +GPIO_UART_UCTS2_PIN +GPIO_UART_URTS2_PIN +[EINT_NAME1] + +[TYPE2] +INFO_0 = UART3 +[GPIO_NAME2] +GPIO_UART_URXD3_PIN +GPIO_UART_UTXD3_PIN +GPIO_UART_UCTS3_PIN +GPIO_UART_URTS3_PIN +[EINT_NAME2] + +[TYPE3] +INFO_0 = UART4 +[GPIO_NAME3] +GPIO_UART_URXD4_PIN +GPIO_UART_UTXD4_PIN +GPIO_UART_UCTS4_PIN +GPIO_UART_URTS4_PIN +[EINT_NAME3] diff --git a/tools/dct/Wizard/USB.cmp b/tools/dct/Wizard/USB.cmp new file mode 100755 index 000000000..8fcdd3daf --- /dev/null +++ b/tools/dct/Wizard/USB.cmp @@ -0,0 +1,29 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = USB OTG Enable/Disable +[INFO_0_CONTENT] +Enable +Disable + +[INFO_0_PROPERTY] +MULTIPLE_CHECK = NO + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = Enable +[GPIO_NAME0] +GPIO_OTG_IDDIG_EINT_PIN +GPIO_OTG_DRVVBUS_PIN +[EINT_NAME0] +OTG_IDDIG + +[TYPE1] +INFO_0 = Disable +[GPIO_NAME1] + +[EINT_NAME1] diff --git a/tools/dct/Wizard/WIFI.cmp b/tools/dct/Wizard/WIFI.cmp new file mode 100755 index 000000000..d9a9e2c09 --- /dev/null +++ b/tools/dct/Wizard/WIFI.cmp @@ -0,0 +1,38 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = Wifi Chip Type +[INFO_0_CONTENT] +MT5921 +MT6620 + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = MT5921 +[GPIO_NAME0] +GPIO_WIFI_RST_PIN +GPIO_WIFI_CLK_PIN +GPIO_WIFI_EINT_PIN +[EINT_NAME0] +MT5921_WIFI + +[TYPE1] +INFO_0 = MT6620 +[GPIO_NAME1] +GPIO_COMBO_6620_LDO_EN_PIN +GPIO_COMBO_PMU_EN_PIN +GPIO_COMBO_RST_PIN +GPIO_COMBO_RTCCLK_PIN +GPIO_COMBO_BGF_EINT_PIN +GPIO_COMBO_ALL_EINT_PIN +[EINT_NAME1] +WIFI + + + + + diff --git a/tools/dct/Wizard/connectivity.cmp b/tools/dct/Wizard/connectivity.cmp new file mode 100755 index 000000000..19ea71afd --- /dev/null +++ b/tools/dct/Wizard/connectivity.cmp @@ -0,0 +1,37 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = I2C +[INFO_0_CONTENT] +I2C0 +I2C1 +I2C2 +[INFO_0_PROPERTY] +MULTIPLE_CHECK = YES + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + + +[TYPE0] +INFO_0 = I2C0 +[GPIO_NAME0] +GPIO_I2C_SCA_PIN +[EINT_NAME0] +MT7118_WIMAX + +[TYPE1] +INFO_0 = I2C1 +[GPIO_NAME1] +GPIO_I2C_SDA_PIN +[EINT_NAME1] + + +[TYPE2] +INFO_0 = I2C2 +[GPIO_NAME2] +GPIO_I2C3_SCA_PIN +[EINT_NAME2] +HALL_2 diff --git a/tools/dct/Wizard/mATV.cmp b/tools/dct/Wizard/mATV.cmp new file mode 100755 index 000000000..5a56baa0d --- /dev/null +++ b/tools/dct/Wizard/mATV.cmp @@ -0,0 +1,25 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = mATV +[INFO_0_CONTENT] +Enable +Disable + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = Enable +[GPIO_NAME0] +GPIO_MATV_PWR_ENABLE +GPIO_MATV_N_RST +[EINT_NAME0] + +[TYPE1] +INFO_0 = Disable +[GPIO_NAME1] + +[EINT_NAME1] diff --git a/tools/dct/Wizard/mATV_old.cmp b/tools/dct/Wizard/mATV_old.cmp new file mode 100755 index 000000000..5a56baa0d --- /dev/null +++ b/tools/dct/Wizard/mATV_old.cmp @@ -0,0 +1,25 @@ +[MODULE_INFO] +INFO_NUM = 1 + +[INFO_0_NAME] +INFO_NAME = mATV +[INFO_0_CONTENT] +Enable +Disable + +[GPIO_PROTECT] +; Pull_en Pull_sel Def_dir In Out Inv Out_High +; dis:0 low:0 out:1 + +[TYPE0] +INFO_0 = Enable +[GPIO_NAME0] +GPIO_MATV_PWR_ENABLE +GPIO_MATV_N_RST +[EINT_NAME0] + +[TYPE1] +INFO_0 = Disable +[GPIO_NAME1] + +[EINT_NAME1] diff --git a/tools/firewire/Makefile b/tools/firewire/Makefile new file mode 100644 index 000000000..81767adaa --- /dev/null +++ b/tools/firewire/Makefile @@ -0,0 +1,19 @@ +prefix = /usr +nosy-dump-version = 0.4 + +CC = gcc + +all : nosy-dump + +nosy-dump : CFLAGS = -Wall -O2 -g +nosy-dump : CPPFLAGS = -DVERSION=\"$(nosy-dump-version)\" -I../../drivers/firewire +nosy-dump : LDFLAGS = -g +nosy-dump : LDLIBS = -lpopt + +nosy-dump : nosy-dump.o decode-fcp.o + +clean : + rm -rf *.o nosy-dump + +install : + install nosy-dump $(prefix)/bin/nosy-dump diff --git a/tools/firewire/decode-fcp.c b/tools/firewire/decode-fcp.c new file mode 100644 index 000000000..e41223b6a --- /dev/null +++ b/tools/firewire/decode-fcp.c @@ -0,0 +1,213 @@ +#include +#include +#include + +#include "list.h" +#include "nosy-dump.h" + +#define CSR_FCP_COMMAND 0xfffff0000b00ull +#define CSR_FCP_RESPONSE 0xfffff0000d00ull + +static const char * const ctype_names[] = { + [0x0] = "control", [0x8] = "not implemented", + [0x1] = "status", [0x9] = "accepted", + [0x2] = "specific inquiry", [0xa] = "rejected", + [0x3] = "notify", [0xb] = "in transition", + [0x4] = "general inquiry", [0xc] = "stable", + [0x5] = "(reserved 0x05)", [0xd] = "changed", + [0x6] = "(reserved 0x06)", [0xe] = "(reserved 0x0e)", + [0x7] = "(reserved 0x07)", [0xf] = "interim", +}; + +static const char * const subunit_type_names[] = { + [0x00] = "monitor", [0x10] = "(reserved 0x10)", + [0x01] = "audio", [0x11] = "(reserved 0x11)", + [0x02] = "printer", [0x12] = "(reserved 0x12)", + [0x03] = "disc", [0x13] = "(reserved 0x13)", + [0x04] = "tape recorder/player",[0x14] = "(reserved 0x14)", + [0x05] = "tuner", [0x15] = "(reserved 0x15)", + [0x06] = "ca", [0x16] = "(reserved 0x16)", + [0x07] = "camera", [0x17] = "(reserved 0x17)", + [0x08] = "(reserved 0x08)", [0x18] = "(reserved 0x18)", + [0x09] = "panel", [0x19] = "(reserved 0x19)", + [0x0a] = "bulletin board", [0x1a] = "(reserved 0x1a)", + [0x0b] = "camera storage", [0x1b] = "(reserved 0x1b)", + [0x0c] = "(reserved 0x0c)", [0x1c] = "vendor unique", + [0x0d] = "(reserved 0x0d)", [0x1d] = "all subunit types", + [0x0e] = "(reserved 0x0e)", [0x1e] = "subunit_type extended to next byte", + [0x0f] = "(reserved 0x0f)", [0x1f] = "unit", +}; + +struct avc_enum { + int value; + const char *name; +}; + +struct avc_field { + const char *name; /* Short name for field. */ + int offset; /* Location of field, specified in bits; */ + /* negative means from end of packet. */ + int width; /* Width of field, 0 means use data_length. */ + struct avc_enum *names; +}; + +struct avc_opcode_info { + const char *name; + struct avc_field fields[8]; +}; + +struct avc_enum power_field_names[] = { + { 0x70, "on" }, + { 0x60, "off" }, + { } +}; + +static const struct avc_opcode_info opcode_info[256] = { + + /* TA Document 1999026 */ + /* AV/C Digital Interface Command Set General Specification 4.0 */ + [0xb2] = { "power", { + { "state", 0, 8, power_field_names } + } + }, + [0x30] = { "unit info", { + { "foo", 0, 8 }, + { "unit_type", 8, 5 }, + { "unit", 13, 3 }, + { "company id", 16, 24 }, + } + }, + [0x31] = { "subunit info" }, + [0x01] = { "reserve" }, + [0xb0] = { "version" }, + [0x00] = { "vendor dependent" }, + [0x02] = { "plug info" }, + [0x12] = { "channel usage" }, + [0x24] = { "connect" }, + [0x20] = { "connect av" }, + [0x22] = { "connections" }, + [0x11] = { "digital input" }, + [0x10] = { "digital output" }, + [0x25] = { "disconnect" }, + [0x21] = { "disconnect av" }, + [0x19] = { "input plug signal format" }, + [0x18] = { "output plug signal format" }, + [0x1f] = { "general bus setup" }, + + /* TA Document 1999025 */ + /* AV/C Descriptor Mechanism Specification Version 1.0 */ + [0x0c] = { "create descriptor" }, + [0x08] = { "open descriptor" }, + [0x09] = { "read descriptor" }, + [0x0a] = { "write descriptor" }, + [0x05] = { "open info block" }, + [0x06] = { "read info block" }, + [0x07] = { "write info block" }, + [0x0b] = { "search descriptor" }, + [0x0d] = { "object number select" }, + + /* TA Document 1999015 */ + /* AV/C Command Set for Rate Control of Isochronous Data Flow 1.0 */ + [0xb3] = { "rate", { + { "subfunction", 0, 8 }, + { "result", 8, 8 }, + { "plug_type", 16, 8 }, + { "plug_id", 16, 8 }, + } + }, + + /* TA Document 1999008 */ + /* AV/C Audio Subunit Specification 1.0 */ + [0xb8] = { "function block" }, + + /* TA Document 2001001 */ + /* AV/C Panel Subunit Specification 1.1 */ + [0x7d] = { "gui update" }, + [0x7e] = { "push gui data" }, + [0x7f] = { "user action" }, + [0x7c] = { "pass through" }, + + /* */ + [0x26] = { "asynchronous connection" }, +}; + +struct avc_frame { + uint32_t operand0:8; + uint32_t opcode:8; + uint32_t subunit_id:3; + uint32_t subunit_type:5; + uint32_t ctype:4; + uint32_t cts:4; +}; + +static void +decode_avc(struct link_transaction *t) +{ + struct avc_frame *frame = + (struct avc_frame *) t->request->packet.write_block.data; + const struct avc_opcode_info *info; + const char *name; + char buffer[32]; + int i; + + info = &opcode_info[frame->opcode]; + if (info->name == NULL) { + snprintf(buffer, sizeof(buffer), + "(unknown opcode 0x%02x)", frame->opcode); + name = buffer; + } else { + name = info->name; + } + + printf("av/c %s, subunit_type=%s, subunit_id=%d, opcode=%s", + ctype_names[frame->ctype], subunit_type_names[frame->subunit_type], + frame->subunit_id, name); + + for (i = 0; info->fields[i].name != NULL; i++) + printf(", %s", info->fields[i].name); + + printf("\n"); +} + +int +decode_fcp(struct link_transaction *t) +{ + struct avc_frame *frame = + (struct avc_frame *) t->request->packet.write_block.data; + unsigned long long offset = + ((unsigned long long) t->request->packet.common.offset_high << 32) | + t->request->packet.common.offset_low; + + if (t->request->packet.common.tcode != TCODE_WRITE_BLOCK_REQUEST) + return 0; + + if (offset == CSR_FCP_COMMAND || offset == CSR_FCP_RESPONSE) { + switch (frame->cts) { + case 0x00: + decode_avc(t); + break; + case 0x01: + printf("cal fcp frame (cts=0x01)\n"); + break; + case 0x02: + printf("ehs fcp frame (cts=0x02)\n"); + break; + case 0x03: + printf("havi fcp frame (cts=0x03)\n"); + break; + case 0x0e: + printf("vendor specific fcp frame (cts=0x0e)\n"); + break; + case 0x0f: + printf("extended cts\n"); + break; + default: + printf("reserved fcp frame (ctx=0x%02x)\n", frame->cts); + break; + } + return 1; + } + + return 0; +} + diff --git a/tools/firewire/list.h b/tools/firewire/list.h new file mode 100644 index 000000000..41f4bdadf --- /dev/null +++ b/tools/firewire/list.h @@ -0,0 +1,62 @@ +struct list { + struct list *next, *prev; +}; + +static inline void +list_init(struct list *list) +{ + list->next = list; + list->prev = list; +} + +static inline int +list_empty(struct list *list) +{ + return list->next == list; +} + +static inline void +list_insert(struct list *link, struct list *new_link) +{ + new_link->prev = link->prev; + new_link->next = link; + new_link->prev->next = new_link; + new_link->next->prev = new_link; +} + +static inline void +list_append(struct list *list, struct list *new_link) +{ + list_insert((struct list *)list, new_link); +} + +static inline void +list_prepend(struct list *list, struct list *new_link) +{ + list_insert(list->next, new_link); +} + +static inline void +list_remove(struct list *link) +{ + link->prev->next = link->next; + link->next->prev = link->prev; +} + +#define list_entry(link, type, member) \ + ((type *)((char *)(link)-(unsigned long)(&((type *)0)->member))) + +#define list_head(list, type, member) \ + list_entry((list)->next, type, member) + +#define list_tail(list, type, member) \ + list_entry((list)->prev, type, member) + +#define list_next(elm, member) \ + list_entry((elm)->member.next, typeof(*elm), member) + +#define list_for_each_entry(pos, list, member) \ + for (pos = list_head(list, typeof(*pos), member); \ + &pos->member != (list); \ + pos = list_next(pos, member)) + diff --git a/tools/firewire/nosy-dump.c b/tools/firewire/nosy-dump.c new file mode 100644 index 000000000..3179c711b --- /dev/null +++ b/tools/firewire/nosy-dump.c @@ -0,0 +1,1035 @@ +/* + * nosy-dump - Interface to snoop mode driver for TI PCILynx 1394 controllers + * Copyright (C) 2002-2006 Kristian Høgsberg + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software Foundation, + * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "list.h" +#include "nosy-dump.h" +#include "nosy-user.h" + +enum { + PACKET_FIELD_DETAIL = 0x01, + PACKET_FIELD_DATA_LENGTH = 0x02, + /* Marks the fields we print in transaction view. */ + PACKET_FIELD_TRANSACTION = 0x04, +}; + +static void print_packet(uint32_t *data, size_t length); +static void decode_link_packet(struct link_packet *packet, size_t length, + int include_flags, int exclude_flags); +static int run = 1; +sig_t sys_sigint_handler; + +static char *option_nosy_device = "/dev/nosy"; +static char *option_view = "packet"; +static char *option_output; +static char *option_input; +static int option_hex; +static int option_iso; +static int option_cycle_start; +static int option_version; +static int option_verbose; + +enum { + VIEW_TRANSACTION, + VIEW_PACKET, + VIEW_STATS, +}; + +static const struct poptOption options[] = { + { + .longName = "device", + .shortName = 'd', + .argInfo = POPT_ARG_STRING, + .arg = &option_nosy_device, + .descrip = "Path to nosy device.", + .argDescrip = "DEVICE" + }, + { + .longName = "view", + .argInfo = POPT_ARG_STRING, + .arg = &option_view, + .descrip = "Specify view of bus traffic: packet, transaction or stats.", + .argDescrip = "VIEW" + }, + { + .longName = "hex", + .shortName = 'x', + .argInfo = POPT_ARG_NONE, + .arg = &option_hex, + .descrip = "Print each packet in hex.", + }, + { + .longName = "iso", + .argInfo = POPT_ARG_NONE, + .arg = &option_iso, + .descrip = "Print iso packets.", + }, + { + .longName = "cycle-start", + .argInfo = POPT_ARG_NONE, + .arg = &option_cycle_start, + .descrip = "Print cycle start packets.", + }, + { + .longName = "verbose", + .shortName = 'v', + .argInfo = POPT_ARG_NONE, + .arg = &option_verbose, + .descrip = "Verbose packet view.", + }, + { + .longName = "output", + .shortName = 'o', + .argInfo = POPT_ARG_STRING, + .arg = &option_output, + .descrip = "Log to output file.", + .argDescrip = "FILENAME" + }, + { + .longName = "input", + .shortName = 'i', + .argInfo = POPT_ARG_STRING, + .arg = &option_input, + .descrip = "Decode log from file.", + .argDescrip = "FILENAME" + }, + { + .longName = "version", + .argInfo = POPT_ARG_NONE, + .arg = &option_version, + .descrip = "Specify print version info.", + }, + POPT_AUTOHELP + POPT_TABLEEND +}; + +/* Allow all ^C except the first to interrupt the program in the usual way. */ +static void +sigint_handler(int signal_num) +{ + if (run == 1) { + run = 0; + signal(SIGINT, SIG_DFL); + } +} + +static struct subaction * +subaction_create(uint32_t *data, size_t length) +{ + struct subaction *sa; + + /* we put the ack in the subaction struct for easy access. */ + sa = malloc(sizeof *sa - sizeof sa->packet + length); + if (!sa) + exit(EXIT_FAILURE); + sa->ack = data[length / 4 - 1]; + sa->length = length; + memcpy(&sa->packet, data, length); + + return sa; +} + +static void +subaction_destroy(struct subaction *sa) +{ + free(sa); +} + +static struct list pending_transaction_list = { + &pending_transaction_list, &pending_transaction_list +}; + +static struct link_transaction * +link_transaction_lookup(int request_node, int response_node, int tlabel) +{ + struct link_transaction *t; + + list_for_each_entry(t, &pending_transaction_list, link) { + if (t->request_node == request_node && + t->response_node == response_node && + t->tlabel == tlabel) + return t; + } + + t = malloc(sizeof *t); + if (!t) + exit(EXIT_FAILURE); + t->request_node = request_node; + t->response_node = response_node; + t->tlabel = tlabel; + list_init(&t->request_list); + list_init(&t->response_list); + + list_append(&pending_transaction_list, &t->link); + + return t; +} + +static void +link_transaction_destroy(struct link_transaction *t) +{ + struct subaction *sa; + + while (!list_empty(&t->request_list)) { + sa = list_head(&t->request_list, struct subaction, link); + list_remove(&sa->link); + subaction_destroy(sa); + } + while (!list_empty(&t->response_list)) { + sa = list_head(&t->response_list, struct subaction, link); + list_remove(&sa->link); + subaction_destroy(sa); + } + free(t); +} + +struct protocol_decoder { + const char *name; + int (*decode)(struct link_transaction *t); +}; + +static const struct protocol_decoder protocol_decoders[] = { + { "FCP", decode_fcp } +}; + +static void +handle_transaction(struct link_transaction *t) +{ + struct subaction *sa; + int i; + + if (!t->request) { + printf("BUG in handle_transaction\n"); + return; + } + + for (i = 0; i < array_length(protocol_decoders); i++) + if (protocol_decoders[i].decode(t)) + break; + + /* HACK: decode only fcp right now. */ + return; + + decode_link_packet(&t->request->packet, t->request->length, + PACKET_FIELD_TRANSACTION, 0); + if (t->response) + decode_link_packet(&t->response->packet, t->request->length, + PACKET_FIELD_TRANSACTION, 0); + else + printf("[no response]"); + + if (option_verbose) { + list_for_each_entry(sa, &t->request_list, link) + print_packet((uint32_t *) &sa->packet, sa->length); + list_for_each_entry(sa, &t->response_list, link) + print_packet((uint32_t *) &sa->packet, sa->length); + } + printf("\r\n"); + + link_transaction_destroy(t); +} + +static void +clear_pending_transaction_list(void) +{ + struct link_transaction *t; + + while (!list_empty(&pending_transaction_list)) { + t = list_head(&pending_transaction_list, + struct link_transaction, link); + list_remove(&t->link); + link_transaction_destroy(t); + /* print unfinished transactions */ + } +} + +static const char * const tcode_names[] = { + [0x0] = "write_quadlet_request", [0x6] = "read_quadlet_response", + [0x1] = "write_block_request", [0x7] = "read_block_response", + [0x2] = "write_response", [0x8] = "cycle_start", + [0x3] = "reserved", [0x9] = "lock_request", + [0x4] = "read_quadlet_request", [0xa] = "iso_data", + [0x5] = "read_block_request", [0xb] = "lock_response", +}; + +static const char * const ack_names[] = { + [0x0] = "no ack", [0x8] = "reserved (0x08)", + [0x1] = "ack_complete", [0x9] = "reserved (0x09)", + [0x2] = "ack_pending", [0xa] = "reserved (0x0a)", + [0x3] = "reserved (0x03)", [0xb] = "reserved (0x0b)", + [0x4] = "ack_busy_x", [0xc] = "reserved (0x0c)", + [0x5] = "ack_busy_a", [0xd] = "ack_data_error", + [0x6] = "ack_busy_b", [0xe] = "ack_type_error", + [0x7] = "reserved (0x07)", [0xf] = "reserved (0x0f)", +}; + +static const char * const rcode_names[] = { + [0x0] = "complete", [0x4] = "conflict_error", + [0x1] = "reserved (0x01)", [0x5] = "data_error", + [0x2] = "reserved (0x02)", [0x6] = "type_error", + [0x3] = "reserved (0x03)", [0x7] = "address_error", +}; + +static const char * const retry_names[] = { + [0x0] = "retry_1", + [0x1] = "retry_x", + [0x2] = "retry_a", + [0x3] = "retry_b", +}; + +enum { + PACKET_RESERVED, + PACKET_REQUEST, + PACKET_RESPONSE, + PACKET_OTHER, +}; + +struct packet_info { + const char *name; + int type; + int response_tcode; + const struct packet_field *fields; + int field_count; +}; + +struct packet_field { + const char *name; /* Short name for field. */ + int offset; /* Location of field, specified in bits; */ + /* negative means from end of packet. */ + int width; /* Width of field, 0 means use data_length. */ + int flags; /* Show options. */ + const char * const *value_names; +}; + +#define COMMON_REQUEST_FIELDS \ + { "dest", 0, 16, PACKET_FIELD_TRANSACTION }, \ + { "tl", 16, 6 }, \ + { "rt", 22, 2, PACKET_FIELD_DETAIL, retry_names }, \ + { "tcode", 24, 4, PACKET_FIELD_TRANSACTION, tcode_names }, \ + { "pri", 28, 4, PACKET_FIELD_DETAIL }, \ + { "src", 32, 16, PACKET_FIELD_TRANSACTION }, \ + { "offs", 48, 48, PACKET_FIELD_TRANSACTION } + +#define COMMON_RESPONSE_FIELDS \ + { "dest", 0, 16 }, \ + { "tl", 16, 6 }, \ + { "rt", 22, 2, PACKET_FIELD_DETAIL, retry_names }, \ + { "tcode", 24, 4, 0, tcode_names }, \ + { "pri", 28, 4, PACKET_FIELD_DETAIL }, \ + { "src", 32, 16 }, \ + { "rcode", 48, 4, PACKET_FIELD_TRANSACTION, rcode_names } + +static const struct packet_field read_quadlet_request_fields[] = { + COMMON_REQUEST_FIELDS, + { "crc", 96, 32, PACKET_FIELD_DETAIL }, + { "ack", 156, 4, 0, ack_names }, +}; + +static const struct packet_field read_quadlet_response_fields[] = { + COMMON_RESPONSE_FIELDS, + { "data", 96, 32, PACKET_FIELD_TRANSACTION }, + { "crc", 128, 32, PACKET_FIELD_DETAIL }, + { "ack", 188, 4, 0, ack_names }, +}; + +static const struct packet_field read_block_request_fields[] = { + COMMON_REQUEST_FIELDS, + { "data_length", 96, 16, PACKET_FIELD_TRANSACTION }, + { "extended_tcode", 112, 16 }, + { "crc", 128, 32, PACKET_FIELD_DETAIL }, + { "ack", 188, 4, 0, ack_names }, +}; + +static const struct packet_field block_response_fields[] = { + COMMON_RESPONSE_FIELDS, + { "data_length", 96, 16, PACKET_FIELD_DATA_LENGTH }, + { "extended_tcode", 112, 16 }, + { "crc", 128, 32, PACKET_FIELD_DETAIL }, + { "data", 160, 0, PACKET_FIELD_TRANSACTION }, + { "crc", -64, 32, PACKET_FIELD_DETAIL }, + { "ack", -4, 4, 0, ack_names }, +}; + +static const struct packet_field write_quadlet_request_fields[] = { + COMMON_REQUEST_FIELDS, + { "data", 96, 32, PACKET_FIELD_TRANSACTION }, + { "ack", -4, 4, 0, ack_names }, +}; + +static const struct packet_field block_request_fields[] = { + COMMON_REQUEST_FIELDS, + { "data_length", 96, 16, PACKET_FIELD_DATA_LENGTH | PACKET_FIELD_TRANSACTION }, + { "extended_tcode", 112, 16, PACKET_FIELD_TRANSACTION }, + { "crc", 128, 32, PACKET_FIELD_DETAIL }, + { "data", 160, 0, PACKET_FIELD_TRANSACTION }, + { "crc", -64, 32, PACKET_FIELD_DETAIL }, + { "ack", -4, 4, 0, ack_names }, +}; + +static const struct packet_field write_response_fields[] = { + COMMON_RESPONSE_FIELDS, + { "reserved", 64, 32, PACKET_FIELD_DETAIL }, + { "ack", -4, 4, 0, ack_names }, +}; + +static const struct packet_field iso_data_fields[] = { + { "data_length", 0, 16, PACKET_FIELD_DATA_LENGTH }, + { "tag", 16, 2 }, + { "channel", 18, 6 }, + { "tcode", 24, 4, 0, tcode_names }, + { "sy", 28, 4 }, + { "crc", 32, 32, PACKET_FIELD_DETAIL }, + { "data", 64, 0 }, + { "crc", -64, 32, PACKET_FIELD_DETAIL }, + { "ack", -4, 4, 0, ack_names }, +}; + +static const struct packet_info packet_info[] = { + { + .name = "write_quadlet_request", + .type = PACKET_REQUEST, + .response_tcode = TCODE_WRITE_RESPONSE, + .fields = write_quadlet_request_fields, + .field_count = array_length(write_quadlet_request_fields) + }, + { + .name = "write_block_request", + .type = PACKET_REQUEST, + .response_tcode = TCODE_WRITE_RESPONSE, + .fields = block_request_fields, + .field_count = array_length(block_request_fields) + }, + { + .name = "write_response", + .type = PACKET_RESPONSE, + .fields = write_response_fields, + .field_count = array_length(write_response_fields) + }, + { + .name = "reserved", + .type = PACKET_RESERVED, + }, + { + .name = "read_quadlet_request", + .type = PACKET_REQUEST, + .response_tcode = TCODE_READ_QUADLET_RESPONSE, + .fields = read_quadlet_request_fields, + .field_count = array_length(read_quadlet_request_fields) + }, + { + .name = "read_block_request", + .type = PACKET_REQUEST, + .response_tcode = TCODE_READ_BLOCK_RESPONSE, + .fields = read_block_request_fields, + .field_count = array_length(read_block_request_fields) + }, + { + .name = "read_quadlet_response", + .type = PACKET_RESPONSE, + .fields = read_quadlet_response_fields, + .field_count = array_length(read_quadlet_response_fields) + }, + { + .name = "read_block_response", + .type = PACKET_RESPONSE, + .fields = block_response_fields, + .field_count = array_length(block_response_fields) + }, + { + .name = "cycle_start", + .type = PACKET_OTHER, + .fields = write_quadlet_request_fields, + .field_count = array_length(write_quadlet_request_fields) + }, + { + .name = "lock_request", + .type = PACKET_REQUEST, + .fields = block_request_fields, + .field_count = array_length(block_request_fields) + }, + { + .name = "iso_data", + .type = PACKET_OTHER, + .fields = iso_data_fields, + .field_count = array_length(iso_data_fields) + }, + { + .name = "lock_response", + .type = PACKET_RESPONSE, + .fields = block_response_fields, + .field_count = array_length(block_response_fields) + }, +}; + +static int +handle_request_packet(uint32_t *data, size_t length) +{ + struct link_packet *p = (struct link_packet *) data; + struct subaction *sa, *prev; + struct link_transaction *t; + + t = link_transaction_lookup(p->common.source, p->common.destination, + p->common.tlabel); + sa = subaction_create(data, length); + t->request = sa; + + if (!list_empty(&t->request_list)) { + prev = list_tail(&t->request_list, + struct subaction, link); + + if (!ACK_BUSY(prev->ack)) { + /* + * error, we should only see ack_busy_* before the + * ack_pending/ack_complete -- this is an ack_pending + * instead (ack_complete would have finished the + * transaction). + */ + } + + if (prev->packet.common.tcode != sa->packet.common.tcode || + prev->packet.common.tlabel != sa->packet.common.tlabel) { + /* memcmp() ? */ + /* error, these should match for retries. */ + } + } + + list_append(&t->request_list, &sa->link); + + switch (sa->ack) { + case ACK_COMPLETE: + if (p->common.tcode != TCODE_WRITE_QUADLET_REQUEST && + p->common.tcode != TCODE_WRITE_BLOCK_REQUEST) + /* error, unified transactions only allowed for write */; + list_remove(&t->link); + handle_transaction(t); + break; + + case ACK_NO_ACK: + case ACK_DATA_ERROR: + case ACK_TYPE_ERROR: + list_remove(&t->link); + handle_transaction(t); + break; + + case ACK_PENDING: + /* request subaction phase over, wait for response. */ + break; + + case ACK_BUSY_X: + case ACK_BUSY_A: + case ACK_BUSY_B: + /* ok, wait for retry. */ + /* check that retry protocol is respected. */ + break; + } + + return 1; +} + +static int +handle_response_packet(uint32_t *data, size_t length) +{ + struct link_packet *p = (struct link_packet *) data; + struct subaction *sa, *prev; + struct link_transaction *t; + + t = link_transaction_lookup(p->common.destination, p->common.source, + p->common.tlabel); + if (list_empty(&t->request_list)) { + /* unsolicited response */ + } + + sa = subaction_create(data, length); + t->response = sa; + + if (!list_empty(&t->response_list)) { + prev = list_tail(&t->response_list, struct subaction, link); + + if (!ACK_BUSY(prev->ack)) { + /* + * error, we should only see ack_busy_* before the + * ack_pending/ack_complete + */ + } + + if (prev->packet.common.tcode != sa->packet.common.tcode || + prev->packet.common.tlabel != sa->packet.common.tlabel) { + /* use memcmp() instead? */ + /* error, these should match for retries. */ + } + } else { + prev = list_tail(&t->request_list, struct subaction, link); + if (prev->ack != ACK_PENDING) { + /* + * error, should not get response unless last request got + * ack_pending. + */ + } + + if (packet_info[prev->packet.common.tcode].response_tcode != + sa->packet.common.tcode) { + /* error, tcode mismatch */ + } + } + + list_append(&t->response_list, &sa->link); + + switch (sa->ack) { + case ACK_COMPLETE: + case ACK_NO_ACK: + case ACK_DATA_ERROR: + case ACK_TYPE_ERROR: + list_remove(&t->link); + handle_transaction(t); + /* transaction complete, remove t from pending list. */ + break; + + case ACK_PENDING: + /* error for responses. */ + break; + + case ACK_BUSY_X: + case ACK_BUSY_A: + case ACK_BUSY_B: + /* no problem, wait for next retry */ + break; + } + + return 1; +} + +static int +handle_packet(uint32_t *data, size_t length) +{ + if (length == 0) { + printf("bus reset\r\n"); + clear_pending_transaction_list(); + } else if (length > sizeof(struct phy_packet)) { + struct link_packet *p = (struct link_packet *) data; + + switch (packet_info[p->common.tcode].type) { + case PACKET_REQUEST: + return handle_request_packet(data, length); + + case PACKET_RESPONSE: + return handle_response_packet(data, length); + + case PACKET_OTHER: + case PACKET_RESERVED: + return 0; + } + } + + return 1; +} + +static unsigned int +get_bits(struct link_packet *packet, int offset, int width) +{ + uint32_t *data = (uint32_t *) packet; + uint32_t index, shift, mask; + + index = offset / 32 + 1; + shift = 32 - (offset & 31) - width; + mask = width == 32 ? ~0 : (1 << width) - 1; + + return (data[index] >> shift) & mask; +} + +#if __BYTE_ORDER == __LITTLE_ENDIAN +#define byte_index(i) ((i) ^ 3) +#elif __BYTE_ORDER == __BIG_ENDIAN +#define byte_index(i) (i) +#else +#error unsupported byte order. +#endif + +static void +dump_data(unsigned char *data, int length) +{ + int i, print_length; + + if (length > 128) + print_length = 128; + else + print_length = length; + + for (i = 0; i < print_length; i++) + printf("%s%02hhx", + (i % 4 == 0 && i != 0) ? " " : "", + data[byte_index(i)]); + + if (print_length < length) + printf(" (%d more bytes)", length - print_length); +} + +static void +decode_link_packet(struct link_packet *packet, size_t length, + int include_flags, int exclude_flags) +{ + const struct packet_info *pi; + int data_length = 0; + int i; + + pi = &packet_info[packet->common.tcode]; + + for (i = 0; i < pi->field_count; i++) { + const struct packet_field *f = &pi->fields[i]; + int offset; + + if (f->flags & exclude_flags) + continue; + if (include_flags && !(f->flags & include_flags)) + continue; + + if (f->offset < 0) + offset = length * 8 + f->offset - 32; + else + offset = f->offset; + + if (f->value_names != NULL) { + uint32_t bits; + + bits = get_bits(packet, offset, f->width); + printf("%s", f->value_names[bits]); + } else if (f->width == 0) { + printf("%s=[", f->name); + dump_data((unsigned char *) packet + (offset / 8 + 4), data_length); + printf("]"); + } else { + unsigned long long bits; + int high_width, low_width; + + if ((offset & ~31) != ((offset + f->width - 1) & ~31)) { + /* Bit field spans quadlet boundary. */ + high_width = ((offset + 31) & ~31) - offset; + low_width = f->width - high_width; + + bits = get_bits(packet, offset, high_width); + bits = (bits << low_width) | + get_bits(packet, offset + high_width, low_width); + } else { + bits = get_bits(packet, offset, f->width); + } + + printf("%s=0x%0*llx", f->name, (f->width + 3) / 4, bits); + + if (f->flags & PACKET_FIELD_DATA_LENGTH) + data_length = bits; + } + + if (i < pi->field_count - 1) + printf(", "); + } +} + +static void +print_packet(uint32_t *data, size_t length) +{ + int i; + + printf("%6u ", data[0]); + + if (length == 4) { + printf("bus reset"); + } else if (length < sizeof(struct phy_packet)) { + printf("short packet: "); + for (i = 1; i < length / 4; i++) + printf("%s%08x", i == 0 ? "[" : " ", data[i]); + printf("]"); + + } else if (length == sizeof(struct phy_packet) && data[1] == ~data[2]) { + struct phy_packet *pp = (struct phy_packet *) data; + + /* phy packet are 3 quadlets: the 1 quadlet payload, + * the bitwise inverse of the payload and the snoop + * mode ack */ + + switch (pp->common.identifier) { + case PHY_PACKET_CONFIGURATION: + if (!pp->phy_config.set_root && !pp->phy_config.set_gap_count) { + printf("ext phy config: phy_id=%02x", pp->phy_config.root_id); + } else { + printf("phy config:"); + if (pp->phy_config.set_root) + printf(" set_root_id=%02x", pp->phy_config.root_id); + if (pp->phy_config.set_gap_count) + printf(" set_gap_count=%d", pp->phy_config.gap_count); + } + break; + + case PHY_PACKET_LINK_ON: + printf("link-on packet, phy_id=%02x", pp->link_on.phy_id); + break; + + case PHY_PACKET_SELF_ID: + if (pp->self_id.extended) { + printf("extended self id: phy_id=%02x, seq=%d", + pp->ext_self_id.phy_id, pp->ext_self_id.sequence); + } else { + static const char * const speed_names[] = { + "S100", "S200", "S400", "BETA" + }; + printf("self id: phy_id=%02x, link %s, gap_count=%d, speed=%s%s%s", + pp->self_id.phy_id, + (pp->self_id.link_active ? "active" : "not active"), + pp->self_id.gap_count, + speed_names[pp->self_id.phy_speed], + (pp->self_id.contender ? ", irm contender" : ""), + (pp->self_id.initiated_reset ? ", initiator" : "")); + } + break; + default: + printf("unknown phy packet: "); + for (i = 1; i < length / 4; i++) + printf("%s%08x", i == 0 ? "[" : " ", data[i]); + printf("]"); + break; + } + } else { + struct link_packet *packet = (struct link_packet *) data; + + decode_link_packet(packet, length, 0, + option_verbose ? 0 : PACKET_FIELD_DETAIL); + } + + if (option_hex) { + printf(" ["); + dump_data((unsigned char *) data + 4, length - 4); + printf("]"); + } + + printf("\r\n"); +} + +#define HIDE_CURSOR "\033[?25l" +#define SHOW_CURSOR "\033[?25h" +#define CLEAR "\033[H\033[2J" + +static void +print_stats(uint32_t *data, size_t length) +{ + static int bus_reset_count, short_packet_count, phy_packet_count; + static int tcode_count[16]; + static struct timeval last_update; + struct timeval now; + int i; + + if (length == 0) + bus_reset_count++; + else if (length < sizeof(struct phy_packet)) + short_packet_count++; + else if (length == sizeof(struct phy_packet) && data[1] == ~data[2]) + phy_packet_count++; + else { + struct link_packet *packet = (struct link_packet *) data; + tcode_count[packet->common.tcode]++; + } + + gettimeofday(&now, NULL); + if (now.tv_sec <= last_update.tv_sec && + now.tv_usec < last_update.tv_usec + 500000) + return; + + last_update = now; + printf(CLEAR HIDE_CURSOR + " bus resets : %8d\n" + " short packets : %8d\n" + " phy packets : %8d\n", + bus_reset_count, short_packet_count, phy_packet_count); + + for (i = 0; i < array_length(packet_info); i++) + if (packet_info[i].type != PACKET_RESERVED) + printf(" %-24s: %8d\n", packet_info[i].name, tcode_count[i]); + printf(SHOW_CURSOR "\n"); +} + +static struct termios saved_attributes; + +static void +reset_input_mode(void) +{ + tcsetattr(STDIN_FILENO, TCSANOW, &saved_attributes); +} + +static void +set_input_mode(void) +{ + struct termios tattr; + + /* Make sure stdin is a terminal. */ + if (!isatty(STDIN_FILENO)) { + fprintf(stderr, "Not a terminal.\n"); + exit(EXIT_FAILURE); + } + + /* Save the terminal attributes so we can restore them later. */ + tcgetattr(STDIN_FILENO, &saved_attributes); + atexit(reset_input_mode); + + /* Set the funny terminal modes. */ + tcgetattr(STDIN_FILENO, &tattr); + tattr.c_lflag &= ~(ICANON|ECHO); /* Clear ICANON and ECHO. */ + tattr.c_cc[VMIN] = 1; + tattr.c_cc[VTIME] = 0; + tcsetattr(STDIN_FILENO, TCSAFLUSH, &tattr); +} + +int main(int argc, const char *argv[]) +{ + uint32_t buf[128 * 1024]; + uint32_t filter; + int length, retval, view; + int fd = -1; + FILE *output = NULL, *input = NULL; + poptContext con; + char c; + struct pollfd pollfds[2]; + + sys_sigint_handler = signal(SIGINT, sigint_handler); + + con = poptGetContext(NULL, argc, argv, options, 0); + retval = poptGetNextOpt(con); + if (retval < -1) { + poptPrintUsage(con, stdout, 0); + return -1; + } + + if (option_version) { + printf("dump tool for nosy sniffer, version %s\n", VERSION); + return 0; + } + + if (__BYTE_ORDER != __LITTLE_ENDIAN) + fprintf(stderr, "warning: nosy has only been tested on little " + "endian machines\n"); + + if (option_input != NULL) { + input = fopen(option_input, "r"); + if (input == NULL) { + fprintf(stderr, "Could not open %s, %m\n", option_input); + return -1; + } + } else { + fd = open(option_nosy_device, O_RDWR); + if (fd < 0) { + fprintf(stderr, "Could not open %s, %m\n", option_nosy_device); + return -1; + } + set_input_mode(); + } + + if (strcmp(option_view, "transaction") == 0) + view = VIEW_TRANSACTION; + else if (strcmp(option_view, "stats") == 0) + view = VIEW_STATS; + else + view = VIEW_PACKET; + + if (option_output) { + output = fopen(option_output, "w"); + if (output == NULL) { + fprintf(stderr, "Could not open %s, %m\n", option_output); + return -1; + } + } + + setvbuf(stdout, NULL, _IOLBF, BUFSIZ); + + filter = ~0; + if (!option_iso) + filter &= ~(1 << TCODE_STREAM_DATA); + if (!option_cycle_start) + filter &= ~(1 << TCODE_CYCLE_START); + if (view == VIEW_STATS) + filter = ~(1 << TCODE_CYCLE_START); + + ioctl(fd, NOSY_IOC_FILTER, filter); + + ioctl(fd, NOSY_IOC_START); + + pollfds[0].fd = fd; + pollfds[0].events = POLLIN; + pollfds[1].fd = STDIN_FILENO; + pollfds[1].events = POLLIN; + + while (run) { + if (input != NULL) { + if (fread(&length, sizeof length, 1, input) != 1) + return 0; + fread(buf, 1, length, input); + } else { + poll(pollfds, 2, -1); + if (pollfds[1].revents) { + read(STDIN_FILENO, &c, sizeof c); + switch (c) { + case 'q': + if (output != NULL) + fclose(output); + return 0; + } + } + + if (pollfds[0].revents) + length = read(fd, buf, sizeof buf); + else + continue; + } + + if (output != NULL) { + fwrite(&length, sizeof length, 1, output); + fwrite(buf, 1, length, output); + } + + switch (view) { + case VIEW_TRANSACTION: + handle_packet(buf, length); + break; + case VIEW_PACKET: + print_packet(buf, length); + break; + case VIEW_STATS: + print_stats(buf, length); + break; + } + } + + if (output != NULL) + fclose(output); + + close(fd); + + poptFreeContext(con); + + return 0; +} diff --git a/tools/firewire/nosy-dump.h b/tools/firewire/nosy-dump.h new file mode 100644 index 000000000..3a4b5b33b --- /dev/null +++ b/tools/firewire/nosy-dump.h @@ -0,0 +1,173 @@ +#ifndef __nosy_dump_h__ +#define __nosy_dump_h__ + +#define array_length(array) (sizeof(array) / sizeof(array[0])) + +#define ACK_NO_ACK 0x0 +#define ACK_DONE(a) ((a >> 2) == 0) +#define ACK_BUSY(a) ((a >> 2) == 1) +#define ACK_ERROR(a) ((a >> 2) == 3) + +#include + +struct phy_packet { + uint32_t timestamp; + union { + struct { + uint32_t zero:24; + uint32_t phy_id:6; + uint32_t identifier:2; + } common, link_on; + + struct { + uint32_t zero:16; + uint32_t gap_count:6; + uint32_t set_gap_count:1; + uint32_t set_root:1; + uint32_t root_id:6; + uint32_t identifier:2; + } phy_config; + + struct { + uint32_t more_packets:1; + uint32_t initiated_reset:1; + uint32_t port2:2; + uint32_t port1:2; + uint32_t port0:2; + uint32_t power_class:3; + uint32_t contender:1; + uint32_t phy_delay:2; + uint32_t phy_speed:2; + uint32_t gap_count:6; + uint32_t link_active:1; + uint32_t extended:1; + uint32_t phy_id:6; + uint32_t identifier:2; + } self_id; + + struct { + uint32_t more_packets:1; + uint32_t reserved1:1; + uint32_t porth:2; + uint32_t portg:2; + uint32_t portf:2; + uint32_t porte:2; + uint32_t portd:2; + uint32_t portc:2; + uint32_t portb:2; + uint32_t porta:2; + uint32_t reserved0:2; + uint32_t sequence:3; + uint32_t extended:1; + uint32_t phy_id:6; + uint32_t identifier:2; + } ext_self_id; + }; + uint32_t inverted; + uint32_t ack; +}; + +#define TCODE_PHY_PACKET 0x10 + +#define PHY_PACKET_CONFIGURATION 0x00 +#define PHY_PACKET_LINK_ON 0x01 +#define PHY_PACKET_SELF_ID 0x02 + +struct link_packet { + uint32_t timestamp; + union { + struct { + uint32_t priority:4; + uint32_t tcode:4; + uint32_t rt:2; + uint32_t tlabel:6; + uint32_t destination:16; + + uint32_t offset_high:16; + uint32_t source:16; + + uint32_t offset_low; + } common; + + struct { + uint32_t common[3]; + uint32_t crc; + } read_quadlet; + + struct { + uint32_t common[3]; + uint32_t data; + uint32_t crc; + } read_quadlet_response; + + struct { + uint32_t common[3]; + uint32_t extended_tcode:16; + uint32_t data_length:16; + uint32_t crc; + } read_block; + + struct { + uint32_t common[3]; + uint32_t extended_tcode:16; + uint32_t data_length:16; + uint32_t crc; + uint32_t data[0]; + /* crc and ack follows. */ + } read_block_response; + + struct { + uint32_t common[3]; + uint32_t data; + uint32_t crc; + } write_quadlet; + + struct { + uint32_t common[3]; + uint32_t extended_tcode:16; + uint32_t data_length:16; + uint32_t crc; + uint32_t data[0]; + /* crc and ack follows. */ + } write_block; + + struct { + uint32_t common[3]; + uint32_t crc; + } write_response; + + struct { + uint32_t common[3]; + uint32_t data; + uint32_t crc; + } cycle_start; + + struct { + uint32_t sy:4; + uint32_t tcode:4; + uint32_t channel:6; + uint32_t tag:2; + uint32_t data_length:16; + + uint32_t crc; + } iso_data; + }; +}; + +struct subaction { + uint32_t ack; + size_t length; + struct list link; + struct link_packet packet; +}; + +struct link_transaction { + int request_node, response_node, tlabel; + struct subaction *request, *response; + struct list request_list, response_list; + struct list link; +}; + +int decode_fcp(struct link_transaction *t); + +#endif /* __nosy_dump_h__ */ diff --git a/tools/hv/hv_get_dhcp_info.sh b/tools/hv/hv_get_dhcp_info.sh new file mode 100755 index 000000000..ccd3e9532 --- /dev/null +++ b/tools/hv/hv_get_dhcp_info.sh @@ -0,0 +1,28 @@ +#!/bin/bash + +# This example script retrieves the DHCP state of a given interface. +# In the interest of keeping the KVP daemon code free of distro specific +# information; the kvp daemon code invokes this external script to gather +# DHCP setting for the specific interface. +# +# Input: Name of the interface +# +# Output: The script prints the string "Enabled" to stdout to indicate +# that DHCP is enabled on the interface. If DHCP is not enabled, +# the script prints the string "Disabled" to stdout. +# +# Each Distro is expected to implement this script in a distro specific +# fashion. For instance on Distros that ship with Network Manager enabled, +# this script can be based on the Network Manager APIs for retrieving DHCP +# information. + +if_file="/etc/sysconfig/network-scripts/ifcfg-"$1 + +dhcp=$(grep "dhcp" $if_file 2>/dev/null) + +if [ "$dhcp" != "" ]; +then +echo "Enabled" +else +echo "Disabled" +fi diff --git a/tools/hv/hv_get_dns_info.sh b/tools/hv/hv_get_dns_info.sh new file mode 100755 index 000000000..058c17b46 --- /dev/null +++ b/tools/hv/hv_get_dns_info.sh @@ -0,0 +1,13 @@ +#!/bin/bash + +# This example script parses /etc/resolv.conf to retrive DNS information. +# In the interest of keeping the KVP daemon code free of distro specific +# information; the kvp daemon code invokes this external script to gather +# DNS information. +# This script is expected to print the nameserver values to stdout. +# Each Distro is expected to implement this script in a distro specific +# fashion. For instance on Distros that ship with Network Manager enabled, +# this script can be based on the Network Manager APIs for retrieving DNS +# entries. + +cat /etc/resolv.conf 2>/dev/null | awk '/^nameserver/ { print $2 }' diff --git a/tools/hv/hv_kvp_daemon.c b/tools/hv/hv_kvp_daemon.c new file mode 100644 index 000000000..274e17867 --- /dev/null +++ b/tools/hv/hv_kvp_daemon.c @@ -0,0 +1,1704 @@ +/* + * An implementation of key value pair (KVP) functionality for Linux. + * + * + * Copyright (C) 2010, Novell, Inc. + * Author : K. Y. Srinivasan + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 as published + * by the Free Software Foundation. + * + * 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, GOOD TITLE or + * NON INFRINGEMENT. See the GNU General Public License for more + * details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + */ + + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +/* + * KVP protocol: The user mode component first registers with the + * the kernel component. Subsequently, the kernel component requests, data + * for the specified keys. In response to this message the user mode component + * fills in the value corresponding to the specified key. We overload the + * sequence field in the cn_msg header to define our KVP message types. + * + * We use this infrastructure for also supporting queries from user mode + * application for state that may be maintained in the KVP kernel component. + * + */ + + +enum key_index { + FullyQualifiedDomainName = 0, + IntegrationServicesVersion, /*This key is serviced in the kernel*/ + NetworkAddressIPv4, + NetworkAddressIPv6, + OSBuildNumber, + OSName, + OSMajorVersion, + OSMinorVersion, + OSVersion, + ProcessorArchitecture +}; + + +enum { + IPADDR = 0, + NETMASK, + GATEWAY, + DNS +}; + +static char kvp_send_buffer[4096]; +static char kvp_recv_buffer[4096 * 2]; +static struct sockaddr_nl addr; +static int in_hand_shake = 1; + +static char *os_name = ""; +static char *os_major = ""; +static char *os_minor = ""; +static char *processor_arch; +static char *os_build; +static char *os_version; +static char *lic_version = "Unknown version"; +static struct utsname uts_buf; + +/* + * The location of the interface configuration file. + */ + +#define KVP_CONFIG_LOC "/var/lib/hyperv" + +#define MAX_FILE_NAME 100 +#define ENTRIES_PER_BLOCK 50 + +#ifndef SOL_NETLINK +#define SOL_NETLINK 270 +#endif + +struct kvp_record { + char key[HV_KVP_EXCHANGE_MAX_KEY_SIZE]; + char value[HV_KVP_EXCHANGE_MAX_VALUE_SIZE]; +}; + +struct kvp_file_state { + int fd; + int num_blocks; + struct kvp_record *records; + int num_records; + char fname[MAX_FILE_NAME]; +}; + +static struct kvp_file_state kvp_file_info[KVP_POOL_COUNT]; + +static void kvp_acquire_lock(int pool) +{ + struct flock fl = {F_WRLCK, SEEK_SET, 0, 0, 0}; + fl.l_pid = getpid(); + + if (fcntl(kvp_file_info[pool].fd, F_SETLKW, &fl) == -1) { + syslog(LOG_ERR, "Failed to acquire the lock pool: %d", pool); + exit(EXIT_FAILURE); + } +} + +static void kvp_release_lock(int pool) +{ + struct flock fl = {F_UNLCK, SEEK_SET, 0, 0, 0}; + fl.l_pid = getpid(); + + if (fcntl(kvp_file_info[pool].fd, F_SETLK, &fl) == -1) { + perror("fcntl"); + syslog(LOG_ERR, "Failed to release the lock pool: %d", pool); + exit(EXIT_FAILURE); + } +} + +static void kvp_update_file(int pool) +{ + FILE *filep; + size_t bytes_written; + + /* + * We are going to write our in-memory registry out to + * disk; acquire the lock first. + */ + kvp_acquire_lock(pool); + + filep = fopen(kvp_file_info[pool].fname, "we"); + if (!filep) { + kvp_release_lock(pool); + syslog(LOG_ERR, "Failed to open file, pool: %d", pool); + exit(EXIT_FAILURE); + } + + bytes_written = fwrite(kvp_file_info[pool].records, + sizeof(struct kvp_record), + kvp_file_info[pool].num_records, filep); + + if (ferror(filep) || fclose(filep)) { + kvp_release_lock(pool); + syslog(LOG_ERR, "Failed to write file, pool: %d", pool); + exit(EXIT_FAILURE); + } + + kvp_release_lock(pool); +} + +static void kvp_update_mem_state(int pool) +{ + FILE *filep; + size_t records_read = 0; + struct kvp_record *record = kvp_file_info[pool].records; + struct kvp_record *readp; + int num_blocks = kvp_file_info[pool].num_blocks; + int alloc_unit = sizeof(struct kvp_record) * ENTRIES_PER_BLOCK; + + kvp_acquire_lock(pool); + + filep = fopen(kvp_file_info[pool].fname, "re"); + if (!filep) { + kvp_release_lock(pool); + syslog(LOG_ERR, "Failed to open file, pool: %d", pool); + exit(EXIT_FAILURE); + } + for (;;) { + readp = &record[records_read]; + records_read += fread(readp, sizeof(struct kvp_record), + ENTRIES_PER_BLOCK * num_blocks, + filep); + + if (ferror(filep)) { + syslog(LOG_ERR, "Failed to read file, pool: %d", pool); + exit(EXIT_FAILURE); + } + + if (!feof(filep)) { + /* + * We have more data to read. + */ + num_blocks++; + record = realloc(record, alloc_unit * num_blocks); + + if (record == NULL) { + syslog(LOG_ERR, "malloc failed"); + exit(EXIT_FAILURE); + } + continue; + } + break; + } + + kvp_file_info[pool].num_blocks = num_blocks; + kvp_file_info[pool].records = record; + kvp_file_info[pool].num_records = records_read; + + fclose(filep); + kvp_release_lock(pool); +} +static int kvp_file_init(void) +{ + int fd; + FILE *filep; + size_t records_read; + char *fname; + struct kvp_record *record; + struct kvp_record *readp; + int num_blocks; + int i; + int alloc_unit = sizeof(struct kvp_record) * ENTRIES_PER_BLOCK; + + if (access(KVP_CONFIG_LOC, F_OK)) { + if (mkdir(KVP_CONFIG_LOC, 0755 /* rwxr-xr-x */)) { + syslog(LOG_ERR, " Failed to create %s", KVP_CONFIG_LOC); + exit(EXIT_FAILURE); + } + } + + for (i = 0; i < KVP_POOL_COUNT; i++) { + fname = kvp_file_info[i].fname; + records_read = 0; + num_blocks = 1; + sprintf(fname, "%s/.kvp_pool_%d", KVP_CONFIG_LOC, i); + fd = open(fname, O_RDWR | O_CREAT | O_CLOEXEC, 0644 /* rw-r--r-- */); + + if (fd == -1) + return 1; + + + filep = fopen(fname, "re"); + if (!filep) + return 1; + + record = malloc(alloc_unit * num_blocks); + if (record == NULL) { + fclose(filep); + return 1; + } + for (;;) { + readp = &record[records_read]; + records_read += fread(readp, sizeof(struct kvp_record), + ENTRIES_PER_BLOCK, + filep); + + if (ferror(filep)) { + syslog(LOG_ERR, "Failed to read file, pool: %d", + i); + exit(EXIT_FAILURE); + } + + if (!feof(filep)) { + /* + * We have more data to read. + */ + num_blocks++; + record = realloc(record, alloc_unit * + num_blocks); + if (record == NULL) { + fclose(filep); + return 1; + } + continue; + } + break; + } + kvp_file_info[i].fd = fd; + kvp_file_info[i].num_blocks = num_blocks; + kvp_file_info[i].records = record; + kvp_file_info[i].num_records = records_read; + fclose(filep); + + } + + return 0; +} + +static int kvp_key_delete(int pool, const char *key, int key_size) +{ + int i; + int j, k; + int num_records; + struct kvp_record *record; + + /* + * First update the in-memory state. + */ + kvp_update_mem_state(pool); + + num_records = kvp_file_info[pool].num_records; + record = kvp_file_info[pool].records; + + for (i = 0; i < num_records; i++) { + if (memcmp(key, record[i].key, key_size)) + continue; + /* + * Found a match; just move the remaining + * entries up. + */ + if (i == num_records) { + kvp_file_info[pool].num_records--; + kvp_update_file(pool); + return 0; + } + + j = i; + k = j + 1; + for (; k < num_records; k++) { + strcpy(record[j].key, record[k].key); + strcpy(record[j].value, record[k].value); + j++; + } + + kvp_file_info[pool].num_records--; + kvp_update_file(pool); + return 0; + } + return 1; +} + +static int kvp_key_add_or_modify(int pool, const char *key, int key_size, const char *value, + int value_size) +{ + int i; + int num_records; + struct kvp_record *record; + int num_blocks; + + if ((key_size > HV_KVP_EXCHANGE_MAX_KEY_SIZE) || + (value_size > HV_KVP_EXCHANGE_MAX_VALUE_SIZE)) + return 1; + + /* + * First update the in-memory state. + */ + kvp_update_mem_state(pool); + + num_records = kvp_file_info[pool].num_records; + record = kvp_file_info[pool].records; + num_blocks = kvp_file_info[pool].num_blocks; + + for (i = 0; i < num_records; i++) { + if (memcmp(key, record[i].key, key_size)) + continue; + /* + * Found a match; just update the value - + * this is the modify case. + */ + memcpy(record[i].value, value, value_size); + kvp_update_file(pool); + return 0; + } + + /* + * Need to add a new entry; + */ + if (num_records == (ENTRIES_PER_BLOCK * num_blocks)) { + /* Need to allocate a larger array for reg entries. */ + record = realloc(record, sizeof(struct kvp_record) * + ENTRIES_PER_BLOCK * (num_blocks + 1)); + + if (record == NULL) + return 1; + kvp_file_info[pool].num_blocks++; + + } + memcpy(record[i].value, value, value_size); + memcpy(record[i].key, key, key_size); + kvp_file_info[pool].records = record; + kvp_file_info[pool].num_records++; + kvp_update_file(pool); + return 0; +} + +static int kvp_get_value(int pool, const char *key, int key_size, char *value, + int value_size) +{ + int i; + int num_records; + struct kvp_record *record; + + if ((key_size > HV_KVP_EXCHANGE_MAX_KEY_SIZE) || + (value_size > HV_KVP_EXCHANGE_MAX_VALUE_SIZE)) + return 1; + + /* + * First update the in-memory state. + */ + kvp_update_mem_state(pool); + + num_records = kvp_file_info[pool].num_records; + record = kvp_file_info[pool].records; + + for (i = 0; i < num_records; i++) { + if (memcmp(key, record[i].key, key_size)) + continue; + /* + * Found a match; just copy the value out. + */ + memcpy(value, record[i].value, value_size); + return 0; + } + + return 1; +} + +static int kvp_pool_enumerate(int pool, int index, char *key, int key_size, + char *value, int value_size) +{ + struct kvp_record *record; + + /* + * First update our in-memory database. + */ + kvp_update_mem_state(pool); + record = kvp_file_info[pool].records; + + if (index >= kvp_file_info[pool].num_records) { + return 1; + } + + memcpy(key, record[index].key, key_size); + memcpy(value, record[index].value, value_size); + return 0; +} + + +void kvp_get_os_info(void) +{ + FILE *file; + char *p, buf[512]; + + uname(&uts_buf); + os_version = uts_buf.release; + os_build = strdup(uts_buf.release); + + os_name = uts_buf.sysname; + processor_arch = uts_buf.machine; + + /* + * The current windows host (win7) expects the build + * string to be of the form: x.y.z + * Strip additional information we may have. + */ + p = strchr(os_version, '-'); + if (p) + *p = '\0'; + + /* + * Parse the /etc/os-release file if present: + * http://www.freedesktop.org/software/systemd/man/os-release.html + */ + file = fopen("/etc/os-release", "r"); + if (file != NULL) { + while (fgets(buf, sizeof(buf), file)) { + char *value, *q; + + /* Ignore comments */ + if (buf[0] == '#') + continue; + + /* Split into name=value */ + p = strchr(buf, '='); + if (!p) + continue; + *p++ = 0; + + /* Remove quotes and newline; un-escape */ + value = p; + q = p; + while (*p) { + if (*p == '\\') { + ++p; + if (!*p) + break; + *q++ = *p++; + } else if (*p == '\'' || *p == '"' || + *p == '\n') { + ++p; + } else { + *q++ = *p++; + } + } + *q = 0; + + if (!strcmp(buf, "NAME")) { + p = strdup(value); + if (!p) + break; + os_name = p; + } else if (!strcmp(buf, "VERSION_ID")) { + p = strdup(value); + if (!p) + break; + os_major = p; + } + } + fclose(file); + return; + } + + /* Fallback for older RH/SUSE releases */ + file = fopen("/etc/SuSE-release", "r"); + if (file != NULL) + goto kvp_osinfo_found; + file = fopen("/etc/redhat-release", "r"); + if (file != NULL) + goto kvp_osinfo_found; + + /* + * We don't have information about the os. + */ + return; + +kvp_osinfo_found: + /* up to three lines */ + p = fgets(buf, sizeof(buf), file); + if (p) { + p = strchr(buf, '\n'); + if (p) + *p = '\0'; + p = strdup(buf); + if (!p) + goto done; + os_name = p; + + /* second line */ + p = fgets(buf, sizeof(buf), file); + if (p) { + p = strchr(buf, '\n'); + if (p) + *p = '\0'; + p = strdup(buf); + if (!p) + goto done; + os_major = p; + + /* third line */ + p = fgets(buf, sizeof(buf), file); + if (p) { + p = strchr(buf, '\n'); + if (p) + *p = '\0'; + p = strdup(buf); + if (p) + os_minor = p; + } + } + } + +done: + fclose(file); + return; +} + + + +/* + * Retrieve an interface name corresponding to the specified guid. + * If there is a match, the function returns a pointer + * to the interface name and if not, a NULL is returned. + * If a match is found, the caller is responsible for + * freeing the memory. + */ + +static char *kvp_get_if_name(char *guid) +{ + DIR *dir; + struct dirent *entry; + FILE *file; + char *p, *q, *x; + char *if_name = NULL; + char buf[256]; + char *kvp_net_dir = "/sys/class/net/"; + char dev_id[256]; + + dir = opendir(kvp_net_dir); + if (dir == NULL) + return NULL; + + snprintf(dev_id, sizeof(dev_id), "%s", kvp_net_dir); + q = dev_id + strlen(kvp_net_dir); + + while ((entry = readdir(dir)) != NULL) { + /* + * Set the state for the next pass. + */ + *q = '\0'; + strcat(dev_id, entry->d_name); + strcat(dev_id, "/device/device_id"); + + file = fopen(dev_id, "r"); + if (file == NULL) + continue; + + p = fgets(buf, sizeof(buf), file); + if (p) { + x = strchr(p, '\n'); + if (x) + *x = '\0'; + + if (!strcmp(p, guid)) { + /* + * Found the guid match; return the interface + * name. The caller will free the memory. + */ + if_name = strdup(entry->d_name); + fclose(file); + break; + } + } + fclose(file); + } + + closedir(dir); + return if_name; +} + +/* + * Retrieve the MAC address given the interface name. + */ + +static char *kvp_if_name_to_mac(char *if_name) +{ + FILE *file; + char *p, *x; + char buf[256]; + char addr_file[256]; + int i; + char *mac_addr = NULL; + + snprintf(addr_file, sizeof(addr_file), "%s%s%s", "/sys/class/net/", + if_name, "/address"); + + file = fopen(addr_file, "r"); + if (file == NULL) + return NULL; + + p = fgets(buf, sizeof(buf), file); + if (p) { + x = strchr(p, '\n'); + if (x) + *x = '\0'; + for (i = 0; i < strlen(p); i++) + p[i] = toupper(p[i]); + mac_addr = strdup(p); + } + + fclose(file); + return mac_addr; +} + + +/* + * Retrieve the interface name given tha MAC address. + */ + +static char *kvp_mac_to_if_name(char *mac) +{ + DIR *dir; + struct dirent *entry; + FILE *file; + char *p, *q, *x; + char *if_name = NULL; + char buf[256]; + char *kvp_net_dir = "/sys/class/net/"; + char dev_id[256]; + int i; + + dir = opendir(kvp_net_dir); + if (dir == NULL) + return NULL; + + snprintf(dev_id, sizeof(dev_id), kvp_net_dir); + q = dev_id + strlen(kvp_net_dir); + + while ((entry = readdir(dir)) != NULL) { + /* + * Set the state for the next pass. + */ + *q = '\0'; + + strcat(dev_id, entry->d_name); + strcat(dev_id, "/address"); + + file = fopen(dev_id, "r"); + if (file == NULL) + continue; + + p = fgets(buf, sizeof(buf), file); + if (p) { + x = strchr(p, '\n'); + if (x) + *x = '\0'; + + for (i = 0; i < strlen(p); i++) + p[i] = toupper(p[i]); + + if (!strcmp(p, mac)) { + /* + * Found the MAC match; return the interface + * name. The caller will free the memory. + */ + if_name = strdup(entry->d_name); + fclose(file); + break; + } + } + fclose(file); + } + + closedir(dir); + return if_name; +} + + +static void kvp_process_ipconfig_file(char *cmd, + char *config_buf, int len, + int element_size, int offset) +{ + char buf[256]; + char *p; + char *x; + FILE *file; + + /* + * First execute the command. + */ + file = popen(cmd, "r"); + if (file == NULL) + return; + + if (offset == 0) + memset(config_buf, 0, len); + while ((p = fgets(buf, sizeof(buf), file)) != NULL) { + if ((len - strlen(config_buf)) < (element_size + 1)) + break; + + x = strchr(p, '\n'); + *x = '\0'; + strcat(config_buf, p); + strcat(config_buf, ";"); + } + pclose(file); +} + +static void kvp_get_ipconfig_info(char *if_name, + struct hv_kvp_ipaddr_value *buffer) +{ + char cmd[512]; + char dhcp_info[128]; + char *p; + FILE *file; + + /* + * Get the address of default gateway (ipv4). + */ + sprintf(cmd, "%s %s", "ip route show dev", if_name); + strcat(cmd, " | awk '/default/ {print $3 }'"); + + /* + * Execute the command to gather gateway info. + */ + kvp_process_ipconfig_file(cmd, (char *)buffer->gate_way, + (MAX_GATEWAY_SIZE * 2), INET_ADDRSTRLEN, 0); + + /* + * Get the address of default gateway (ipv6). + */ + sprintf(cmd, "%s %s", "ip -f inet6 route show dev", if_name); + strcat(cmd, " | awk '/default/ {print $3 }'"); + + /* + * Execute the command to gather gateway info (ipv6). + */ + kvp_process_ipconfig_file(cmd, (char *)buffer->gate_way, + (MAX_GATEWAY_SIZE * 2), INET6_ADDRSTRLEN, 1); + + + /* + * Gather the DNS state. + * Since there is no standard way to get this information + * across various distributions of interest; we just invoke + * an external script that needs to be ported across distros + * of interest. + * + * Following is the expected format of the information from the script: + * + * ipaddr1 (nameserver1) + * ipaddr2 (nameserver2) + * . + * . + */ + + sprintf(cmd, "%s", "hv_get_dns_info"); + + /* + * Execute the command to gather DNS info. + */ + kvp_process_ipconfig_file(cmd, (char *)buffer->dns_addr, + (MAX_IP_ADDR_SIZE * 2), INET_ADDRSTRLEN, 0); + + /* + * Gather the DHCP state. + * We will gather this state by invoking an external script. + * The parameter to the script is the interface name. + * Here is the expected output: + * + * Enabled: DHCP enabled. + */ + + sprintf(cmd, "%s %s", "hv_get_dhcp_info", if_name); + + file = popen(cmd, "r"); + if (file == NULL) + return; + + p = fgets(dhcp_info, sizeof(dhcp_info), file); + if (p == NULL) { + pclose(file); + return; + } + + if (!strncmp(p, "Enabled", 7)) + buffer->dhcp_enabled = 1; + else + buffer->dhcp_enabled = 0; + + pclose(file); +} + + +static unsigned int hweight32(unsigned int *w) +{ + unsigned int res = *w - ((*w >> 1) & 0x55555555); + res = (res & 0x33333333) + ((res >> 2) & 0x33333333); + res = (res + (res >> 4)) & 0x0F0F0F0F; + res = res + (res >> 8); + return (res + (res >> 16)) & 0x000000FF; +} + +static int kvp_process_ip_address(void *addrp, + int family, char *buffer, + int length, int *offset) +{ + struct sockaddr_in *addr; + struct sockaddr_in6 *addr6; + int addr_length; + char tmp[50]; + const char *str; + + if (family == AF_INET) { + addr = (struct sockaddr_in *)addrp; + str = inet_ntop(family, &addr->sin_addr, tmp, 50); + addr_length = INET_ADDRSTRLEN; + } else { + addr6 = (struct sockaddr_in6 *)addrp; + str = inet_ntop(family, &addr6->sin6_addr.s6_addr, tmp, 50); + addr_length = INET6_ADDRSTRLEN; + } + + if ((length - *offset) < addr_length + 2) + return HV_E_FAIL; + if (str == NULL) { + strcpy(buffer, "inet_ntop failed\n"); + return HV_E_FAIL; + } + if (*offset == 0) + strcpy(buffer, tmp); + else { + strcat(buffer, ";"); + strcat(buffer, tmp); + } + + *offset += strlen(str) + 1; + + return 0; +} + +static int +kvp_get_ip_info(int family, char *if_name, int op, + void *out_buffer, int length) +{ + struct ifaddrs *ifap; + struct ifaddrs *curp; + int offset = 0; + int sn_offset = 0; + int error = 0; + char *buffer; + struct hv_kvp_ipaddr_value *ip_buffer; + char cidr_mask[5]; /* /xyz */ + int weight; + int i; + unsigned int *w; + char *sn_str; + struct sockaddr_in6 *addr6; + + if (op == KVP_OP_ENUMERATE) { + buffer = out_buffer; + } else { + ip_buffer = out_buffer; + buffer = (char *)ip_buffer->ip_addr; + ip_buffer->addr_family = 0; + } + /* + * On entry into this function, the buffer is capable of holding the + * maximum key value. + */ + + if (getifaddrs(&ifap)) { + strcpy(buffer, "getifaddrs failed\n"); + return HV_E_FAIL; + } + + curp = ifap; + while (curp != NULL) { + if (curp->ifa_addr == NULL) { + curp = curp->ifa_next; + continue; + } + + if ((if_name != NULL) && + (strncmp(curp->ifa_name, if_name, strlen(if_name)))) { + /* + * We want info about a specific interface; + * just continue. + */ + curp = curp->ifa_next; + continue; + } + + /* + * We only support two address families: AF_INET and AF_INET6. + * If a family value of 0 is specified, we collect both + * supported address families; if not we gather info on + * the specified address family. + */ + if ((((family != 0) && + (curp->ifa_addr->sa_family != family))) || + (curp->ifa_flags & IFF_LOOPBACK)) { + curp = curp->ifa_next; + continue; + } + if ((curp->ifa_addr->sa_family != AF_INET) && + (curp->ifa_addr->sa_family != AF_INET6)) { + curp = curp->ifa_next; + continue; + } + + if (op == KVP_OP_GET_IP_INFO) { + /* + * Gather info other than the IP address. + * IP address info will be gathered later. + */ + if (curp->ifa_addr->sa_family == AF_INET) { + ip_buffer->addr_family |= ADDR_FAMILY_IPV4; + /* + * Get subnet info. + */ + error = kvp_process_ip_address( + curp->ifa_netmask, + AF_INET, + (char *) + ip_buffer->sub_net, + length, + &sn_offset); + if (error) + goto gather_ipaddr; + } else { + ip_buffer->addr_family |= ADDR_FAMILY_IPV6; + + /* + * Get subnet info in CIDR format. + */ + weight = 0; + sn_str = (char *)ip_buffer->sub_net; + addr6 = (struct sockaddr_in6 *) + curp->ifa_netmask; + w = addr6->sin6_addr.s6_addr32; + + for (i = 0; i < 4; i++) + weight += hweight32(&w[i]); + + sprintf(cidr_mask, "/%d", weight); + if ((length - sn_offset) < + (strlen(cidr_mask) + 1)) + goto gather_ipaddr; + + if (sn_offset == 0) + strcpy(sn_str, cidr_mask); + else { + strcat((char *)ip_buffer->sub_net, ";"); + strcat(sn_str, cidr_mask); + } + sn_offset += strlen(sn_str) + 1; + } + + /* + * Collect other ip related configuration info. + */ + + kvp_get_ipconfig_info(if_name, ip_buffer); + } + +gather_ipaddr: + error = kvp_process_ip_address(curp->ifa_addr, + curp->ifa_addr->sa_family, + buffer, + length, &offset); + if (error) + goto getaddr_done; + + curp = curp->ifa_next; + } + +getaddr_done: + freeifaddrs(ifap); + return error; +} + + +static int expand_ipv6(char *addr, int type) +{ + int ret; + struct in6_addr v6_addr; + + ret = inet_pton(AF_INET6, addr, &v6_addr); + + if (ret != 1) { + if (type == NETMASK) + return 1; + return 0; + } + + sprintf(addr, "%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:" + "%02x%02x:%02x%02x:%02x%02x", + (int)v6_addr.s6_addr[0], (int)v6_addr.s6_addr[1], + (int)v6_addr.s6_addr[2], (int)v6_addr.s6_addr[3], + (int)v6_addr.s6_addr[4], (int)v6_addr.s6_addr[5], + (int)v6_addr.s6_addr[6], (int)v6_addr.s6_addr[7], + (int)v6_addr.s6_addr[8], (int)v6_addr.s6_addr[9], + (int)v6_addr.s6_addr[10], (int)v6_addr.s6_addr[11], + (int)v6_addr.s6_addr[12], (int)v6_addr.s6_addr[13], + (int)v6_addr.s6_addr[14], (int)v6_addr.s6_addr[15]); + + return 1; + +} + +static int is_ipv4(char *addr) +{ + int ret; + struct in_addr ipv4_addr; + + ret = inet_pton(AF_INET, addr, &ipv4_addr); + + if (ret == 1) + return 1; + return 0; +} + +static int parse_ip_val_buffer(char *in_buf, int *offset, + char *out_buf, int out_len) +{ + char *x; + char *start; + + /* + * in_buf has sequence of characters that are seperated by + * the character ';'. The last sequence does not have the + * terminating ";" character. + */ + start = in_buf + *offset; + + x = strchr(start, ';'); + if (x) + *x = 0; + else + x = start + strlen(start); + + if (strlen(start) != 0) { + int i = 0; + /* + * Get rid of leading spaces. + */ + while (start[i] == ' ') + i++; + + if ((x - start) <= out_len) { + strcpy(out_buf, (start + i)); + *offset += (x - start) + 1; + return 1; + } + } + return 0; +} + +static int kvp_write_file(FILE *f, char *s1, char *s2, char *s3) +{ + int ret; + + ret = fprintf(f, "%s%s%s%s\n", s1, s2, "=", s3); + + if (ret < 0) + return HV_E_FAIL; + + return 0; +} + + +static int process_ip_string(FILE *f, char *ip_string, int type) +{ + int error = 0; + char addr[INET6_ADDRSTRLEN]; + int i = 0; + int j = 0; + char str[256]; + char sub_str[10]; + int offset = 0; + + memset(addr, 0, sizeof(addr)); + + while (parse_ip_val_buffer(ip_string, &offset, addr, + (MAX_IP_ADDR_SIZE * 2))) { + + sub_str[0] = 0; + if (is_ipv4(addr)) { + switch (type) { + case IPADDR: + snprintf(str, sizeof(str), "%s", "IPADDR"); + break; + case NETMASK: + snprintf(str, sizeof(str), "%s", "NETMASK"); + break; + case GATEWAY: + snprintf(str, sizeof(str), "%s", "GATEWAY"); + break; + case DNS: + snprintf(str, sizeof(str), "%s", "DNS"); + break; + } + + if (type == DNS) { + snprintf(sub_str, sizeof(sub_str), "%d", ++i); + } else if (type == GATEWAY && i == 0) { + ++i; + } else { + snprintf(sub_str, sizeof(sub_str), "%d", i++); + } + + + } else if (expand_ipv6(addr, type)) { + switch (type) { + case IPADDR: + snprintf(str, sizeof(str), "%s", "IPV6ADDR"); + break; + case NETMASK: + snprintf(str, sizeof(str), "%s", "IPV6NETMASK"); + break; + case GATEWAY: + snprintf(str, sizeof(str), "%s", + "IPV6_DEFAULTGW"); + break; + case DNS: + snprintf(str, sizeof(str), "%s", "DNS"); + break; + } + + if (type == DNS) { + snprintf(sub_str, sizeof(sub_str), "%d", ++i); + } else if (j == 0) { + ++j; + } else { + snprintf(sub_str, sizeof(sub_str), "_%d", j++); + } + } else { + return HV_INVALIDARG; + } + + error = kvp_write_file(f, str, sub_str, addr); + if (error) + return error; + memset(addr, 0, sizeof(addr)); + } + + return 0; +} + +static int kvp_set_ip_info(char *if_name, struct hv_kvp_ipaddr_value *new_val) +{ + int error = 0; + char if_file[128]; + FILE *file; + char cmd[512]; + char *mac_addr; + + /* + * Set the configuration for the specified interface with + * the information provided. Since there is no standard + * way to configure an interface, we will have an external + * script that does the job of configuring the interface and + * flushing the configuration. + * + * The parameters passed to this external script are: + * 1. A configuration file that has the specified configuration. + * + * We will embed the name of the interface in the configuration + * file: ifcfg-ethx (where ethx is the interface name). + * + * The information provided here may be more than what is needed + * in a given distro to configure the interface and so are free + * ignore information that may not be relevant. + * + * Here is the format of the ip configuration file: + * + * HWADDR=macaddr + * DEVICE=interface name + * BOOTPROTO= (where is "dhcp" if DHCP is configured + * or "none" if no boot-time protocol should be used) + * + * IPADDR0=ipaddr1 + * IPADDR1=ipaddr2 + * IPADDRx=ipaddry (where y = x + 1) + * + * NETMASK0=netmask1 + * NETMASKx=netmasky (where y = x + 1) + * + * GATEWAY=ipaddr1 + * GATEWAYx=ipaddry (where y = x + 1) + * + * DNSx=ipaddrx (where first DNS address is tagged as DNS1 etc) + * + * IPV6 addresses will be tagged as IPV6ADDR, IPV6 gateway will be + * tagged as IPV6_DEFAULTGW and IPV6 NETMASK will be tagged as + * IPV6NETMASK. + * + * The host can specify multiple ipv4 and ipv6 addresses to be + * configured for the interface. Furthermore, the configuration + * needs to be persistent. A subsequent GET call on the interface + * is expected to return the configuration that is set via the SET + * call. + */ + + snprintf(if_file, sizeof(if_file), "%s%s%s", KVP_CONFIG_LOC, + "/ifcfg-", if_name); + + file = fopen(if_file, "w"); + + if (file == NULL) { + syslog(LOG_ERR, "Failed to open config file"); + return HV_E_FAIL; + } + + /* + * First write out the MAC address. + */ + + mac_addr = kvp_if_name_to_mac(if_name); + if (mac_addr == NULL) { + error = HV_E_FAIL; + goto setval_error; + } + + error = kvp_write_file(file, "HWADDR", "", mac_addr); + if (error) + goto setval_error; + + error = kvp_write_file(file, "DEVICE", "", if_name); + if (error) + goto setval_error; + + if (new_val->dhcp_enabled) { + error = kvp_write_file(file, "BOOTPROTO", "", "dhcp"); + if (error) + goto setval_error; + + /* + * We are done!. + */ + goto setval_done; + + } else { + error = kvp_write_file(file, "BOOTPROTO", "", "none"); + if (error) + goto setval_error; + } + + /* + * Write the configuration for ipaddress, netmask, gateway and + * name servers. + */ + + error = process_ip_string(file, (char *)new_val->ip_addr, IPADDR); + if (error) + goto setval_error; + + error = process_ip_string(file, (char *)new_val->sub_net, NETMASK); + if (error) + goto setval_error; + + error = process_ip_string(file, (char *)new_val->gate_way, GATEWAY); + if (error) + goto setval_error; + + error = process_ip_string(file, (char *)new_val->dns_addr, DNS); + if (error) + goto setval_error; + +setval_done: + free(mac_addr); + fclose(file); + + /* + * Now that we have populated the configuration file, + * invoke the external script to do its magic. + */ + + snprintf(cmd, sizeof(cmd), "%s %s", "hv_set_ifconfig", if_file); + system(cmd); + return 0; + +setval_error: + syslog(LOG_ERR, "Failed to write config file"); + free(mac_addr); + fclose(file); + return error; +} + + +static int +kvp_get_domain_name(char *buffer, int length) +{ + struct addrinfo hints, *info ; + int error = 0; + + gethostname(buffer, length); + memset(&hints, 0, sizeof(hints)); + hints.ai_family = AF_INET; /*Get only ipv4 addrinfo. */ + hints.ai_socktype = SOCK_STREAM; + hints.ai_flags = AI_CANONNAME; + + error = getaddrinfo(buffer, NULL, &hints, &info); + if (error != 0) { + strcpy(buffer, "getaddrinfo failed\n"); + return error; + } + strcpy(buffer, info->ai_canonname); + freeaddrinfo(info); + return error; +} + +static int +netlink_send(int fd, struct cn_msg *msg) +{ + struct nlmsghdr *nlh; + unsigned int size; + struct msghdr message; + char buffer[64]; + struct iovec iov[2]; + + size = NLMSG_SPACE(sizeof(struct cn_msg) + msg->len); + + nlh = (struct nlmsghdr *)buffer; + nlh->nlmsg_seq = 0; + nlh->nlmsg_pid = getpid(); + nlh->nlmsg_type = NLMSG_DONE; + nlh->nlmsg_len = NLMSG_LENGTH(size - sizeof(*nlh)); + nlh->nlmsg_flags = 0; + + iov[0].iov_base = nlh; + iov[0].iov_len = sizeof(*nlh); + + iov[1].iov_base = msg; + iov[1].iov_len = size; + + memset(&message, 0, sizeof(message)); + message.msg_name = &addr; + message.msg_namelen = sizeof(addr); + message.msg_iov = iov; + message.msg_iovlen = 2; + + return sendmsg(fd, &message, 0); +} + +int main(void) +{ + int fd, len, nl_group; + int error; + struct cn_msg *message; + struct pollfd pfd; + struct nlmsghdr *incoming_msg; + struct cn_msg *incoming_cn_msg; + struct hv_kvp_msg *hv_msg; + char *p; + char *key_value; + char *key_name; + int op; + int pool; + char *if_name; + struct hv_kvp_ipaddr_value *kvp_ip_val; + + daemon(1, 0); + openlog("KVP", 0, LOG_USER); + syslog(LOG_INFO, "KVP starting; pid is:%d", getpid()); + /* + * Retrieve OS release information. + */ + kvp_get_os_info(); + + if (kvp_file_init()) { + syslog(LOG_ERR, "Failed to initialize the pools"); + exit(EXIT_FAILURE); + } + + fd = socket(AF_NETLINK, SOCK_DGRAM, NETLINK_CONNECTOR); + if (fd < 0) { + syslog(LOG_ERR, "netlink socket creation failed; error:%d", fd); + exit(EXIT_FAILURE); + } + addr.nl_family = AF_NETLINK; + addr.nl_pad = 0; + addr.nl_pid = 0; + addr.nl_groups = 0; + + + error = bind(fd, (struct sockaddr *)&addr, sizeof(addr)); + if (error < 0) { + syslog(LOG_ERR, "bind failed; error:%d", error); + close(fd); + exit(EXIT_FAILURE); + } + nl_group = CN_KVP_IDX; + setsockopt(fd, SOL_NETLINK, NETLINK_ADD_MEMBERSHIP, &nl_group, sizeof(nl_group)); + /* + * Register ourselves with the kernel. + */ + message = (struct cn_msg *)kvp_send_buffer; + message->id.idx = CN_KVP_IDX; + message->id.val = CN_KVP_VAL; + + hv_msg = (struct hv_kvp_msg *)message->data; + hv_msg->kvp_hdr.operation = KVP_OP_REGISTER1; + message->ack = 0; + message->len = sizeof(struct hv_kvp_msg); + + len = netlink_send(fd, message); + if (len < 0) { + syslog(LOG_ERR, "netlink_send failed; error:%d", len); + close(fd); + exit(EXIT_FAILURE); + } + + pfd.fd = fd; + + while (1) { + struct sockaddr *addr_p = (struct sockaddr *) &addr; + socklen_t addr_l = sizeof(addr); + pfd.events = POLLIN; + pfd.revents = 0; + poll(&pfd, 1, -1); + + len = recvfrom(fd, kvp_recv_buffer, sizeof(kvp_recv_buffer), 0, + addr_p, &addr_l); + + if (len < 0) { + syslog(LOG_ERR, "recvfrom failed; pid:%u error:%d %s", + addr.nl_pid, errno, strerror(errno)); + close(fd); + return -1; + } + + if (addr.nl_pid) { + syslog(LOG_WARNING, "Received packet from untrusted pid:%u", + addr.nl_pid); + continue; + } + + incoming_msg = (struct nlmsghdr *)kvp_recv_buffer; + + if (incoming_msg->nlmsg_type != NLMSG_DONE) + continue; + + incoming_cn_msg = (struct cn_msg *)NLMSG_DATA(incoming_msg); + hv_msg = (struct hv_kvp_msg *)incoming_cn_msg->data; + + /* + * We will use the KVP header information to pass back + * the error from this daemon. So, first copy the state + * and set the error code to success. + */ + op = hv_msg->kvp_hdr.operation; + pool = hv_msg->kvp_hdr.pool; + hv_msg->error = HV_S_OK; + + if ((in_hand_shake) && (op == KVP_OP_REGISTER1)) { + /* + * Driver is registering with us; stash away the version + * information. + */ + in_hand_shake = 0; + p = (char *)hv_msg->body.kvp_register.version; + lic_version = malloc(strlen(p) + 1); + if (lic_version) { + strcpy(lic_version, p); + syslog(LOG_INFO, "KVP LIC Version: %s", + lic_version); + } else { + syslog(LOG_ERR, "malloc failed"); + } + continue; + } + + switch (op) { + case KVP_OP_GET_IP_INFO: + kvp_ip_val = &hv_msg->body.kvp_ip_val; + if_name = + kvp_mac_to_if_name((char *)kvp_ip_val->adapter_id); + + if (if_name == NULL) { + /* + * We could not map the mac address to an + * interface name; return error. + */ + hv_msg->error = HV_E_FAIL; + break; + } + error = kvp_get_ip_info( + 0, if_name, KVP_OP_GET_IP_INFO, + kvp_ip_val, + (MAX_IP_ADDR_SIZE * 2)); + + if (error) + hv_msg->error = error; + + free(if_name); + break; + + case KVP_OP_SET_IP_INFO: + kvp_ip_val = &hv_msg->body.kvp_ip_val; + if_name = kvp_get_if_name( + (char *)kvp_ip_val->adapter_id); + if (if_name == NULL) { + /* + * We could not map the guid to an + * interface name; return error. + */ + hv_msg->error = HV_GUID_NOTFOUND; + break; + } + error = kvp_set_ip_info(if_name, kvp_ip_val); + if (error) + hv_msg->error = error; + + free(if_name); + break; + + case KVP_OP_SET: + if (kvp_key_add_or_modify(pool, + hv_msg->body.kvp_set.data.key, + hv_msg->body.kvp_set.data.key_size, + hv_msg->body.kvp_set.data.value, + hv_msg->body.kvp_set.data.value_size)) + hv_msg->error = HV_S_CONT; + break; + + case KVP_OP_GET: + if (kvp_get_value(pool, + hv_msg->body.kvp_set.data.key, + hv_msg->body.kvp_set.data.key_size, + hv_msg->body.kvp_set.data.value, + hv_msg->body.kvp_set.data.value_size)) + hv_msg->error = HV_S_CONT; + break; + + case KVP_OP_DELETE: + if (kvp_key_delete(pool, + hv_msg->body.kvp_delete.key, + hv_msg->body.kvp_delete.key_size)) + hv_msg->error = HV_S_CONT; + break; + + default: + break; + } + + if (op != KVP_OP_ENUMERATE) + goto kvp_done; + + /* + * If the pool is KVP_POOL_AUTO, dynamically generate + * both the key and the value; if not read from the + * appropriate pool. + */ + if (pool != KVP_POOL_AUTO) { + if (kvp_pool_enumerate(pool, + hv_msg->body.kvp_enum_data.index, + hv_msg->body.kvp_enum_data.data.key, + HV_KVP_EXCHANGE_MAX_KEY_SIZE, + hv_msg->body.kvp_enum_data.data.value, + HV_KVP_EXCHANGE_MAX_VALUE_SIZE)) + hv_msg->error = HV_S_CONT; + goto kvp_done; + } + + hv_msg = (struct hv_kvp_msg *)incoming_cn_msg->data; + key_name = (char *)hv_msg->body.kvp_enum_data.data.key; + key_value = (char *)hv_msg->body.kvp_enum_data.data.value; + + switch (hv_msg->body.kvp_enum_data.index) { + case FullyQualifiedDomainName: + kvp_get_domain_name(key_value, + HV_KVP_EXCHANGE_MAX_VALUE_SIZE); + strcpy(key_name, "FullyQualifiedDomainName"); + break; + case IntegrationServicesVersion: + strcpy(key_name, "IntegrationServicesVersion"); + strcpy(key_value, lic_version); + break; + case NetworkAddressIPv4: + kvp_get_ip_info(AF_INET, NULL, KVP_OP_ENUMERATE, + key_value, HV_KVP_EXCHANGE_MAX_VALUE_SIZE); + strcpy(key_name, "NetworkAddressIPv4"); + break; + case NetworkAddressIPv6: + kvp_get_ip_info(AF_INET6, NULL, KVP_OP_ENUMERATE, + key_value, HV_KVP_EXCHANGE_MAX_VALUE_SIZE); + strcpy(key_name, "NetworkAddressIPv6"); + break; + case OSBuildNumber: + strcpy(key_value, os_build); + strcpy(key_name, "OSBuildNumber"); + break; + case OSName: + strcpy(key_value, os_name); + strcpy(key_name, "OSName"); + break; + case OSMajorVersion: + strcpy(key_value, os_major); + strcpy(key_name, "OSMajorVersion"); + break; + case OSMinorVersion: + strcpy(key_value, os_minor); + strcpy(key_name, "OSMinorVersion"); + break; + case OSVersion: + strcpy(key_value, os_version); + strcpy(key_name, "OSVersion"); + break; + case ProcessorArchitecture: + strcpy(key_value, processor_arch); + strcpy(key_name, "ProcessorArchitecture"); + break; + default: + hv_msg->error = HV_S_CONT; + break; + } + /* + * Send the value back to the kernel. The response is + * already in the receive buffer. Update the cn_msg header to + * reflect the key value that has been added to the message + */ +kvp_done: + + incoming_cn_msg->id.idx = CN_KVP_IDX; + incoming_cn_msg->id.val = CN_KVP_VAL; + incoming_cn_msg->ack = 0; + incoming_cn_msg->len = sizeof(struct hv_kvp_msg); + + len = netlink_send(fd, incoming_cn_msg); + if (len < 0) { + syslog(LOG_ERR, "net_link send failed; error:%d", len); + exit(EXIT_FAILURE); + } + } + +} diff --git a/tools/hv/hv_set_ifconfig.sh b/tools/hv/hv_set_ifconfig.sh new file mode 100755 index 000000000..735aafd64 --- /dev/null +++ b/tools/hv/hv_set_ifconfig.sh @@ -0,0 +1,64 @@ +#!/bin/bash + +# This example script activates an interface based on the specified +# configuration. +# +# In the interest of keeping the KVP daemon code free of distro specific +# information; the kvp daemon code invokes this external script to configure +# the interface. +# +# The only argument to this script is the configuration file that is to +# be used to configure the interface. +# +# Each Distro is expected to implement this script in a distro specific +# fashion. For instance on Distros that ship with Network Manager enabled, +# this script can be based on the Network Manager APIs for configuring the +# interface. +# +# This example script is based on a RHEL environment. +# +# Here is the format of the ip configuration file: +# +# HWADDR=macaddr +# DEVICE=interface name +# BOOTPROTO= (where is "dhcp" if DHCP is configured +# or "none" if no boot-time protocol should be used) +# +# IPADDR0=ipaddr1 +# IPADDR1=ipaddr2 +# IPADDRx=ipaddry (where y = x + 1) +# +# NETMASK0=netmask1 +# NETMASKx=netmasky (where y = x + 1) +# +# GATEWAY=ipaddr1 +# GATEWAYx=ipaddry (where y = x + 1) +# +# DNSx=ipaddrx (where first DNS address is tagged as DNS1 etc) +# +# IPV6 addresses will be tagged as IPV6ADDR, IPV6 gateway will be +# tagged as IPV6_DEFAULTGW and IPV6 NETMASK will be tagged as +# IPV6NETMASK. +# +# The host can specify multiple ipv4 and ipv6 addresses to be +# configured for the interface. Furthermore, the configuration +# needs to be persistent. A subsequent GET call on the interface +# is expected to return the configuration that is set via the SET +# call. +# + + + +echo "IPV6INIT=yes" >> $1 +echo "NM_CONTROLLED=no" >> $1 +echo "PEERDNS=yes" >> $1 +echo "ONBOOT=yes" >> $1 + + +cp $1 /etc/sysconfig/network-scripts/ + + +interface=$(echo $1 | awk -F - '{ print $2 }') + +/sbin/ifdown $interface 2>/dev/null +/sbin/ifup $interface 2>/dev/null diff --git a/tools/hv/hv_vss_daemon.c b/tools/hv/hv_vss_daemon.c new file mode 100644 index 000000000..fea03a3ed --- /dev/null +++ b/tools/hv/hv_vss_daemon.c @@ -0,0 +1,249 @@ +/* + * An implementation of the host initiated guest snapshot for Hyper-V. + * + * + * Copyright (C) 2013, Microsoft, Inc. + * Author : K. Y. Srinivasan + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 as published + * by the Free Software Foundation. + * + * 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, GOOD TITLE or + * NON INFRINGEMENT. See the GNU General Public License for more + * details. + * + */ + + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +static char vss_recv_buffer[4096]; +static char vss_send_buffer[4096]; +static struct sockaddr_nl addr; + +#ifndef SOL_NETLINK +#define SOL_NETLINK 270 +#endif + + +static int vss_do_freeze(char *dir, unsigned int cmd, char *fs_op) +{ + int ret, fd = open(dir, O_RDONLY); + + if (fd < 0) + return 1; + ret = ioctl(fd, cmd, 0); + syslog(LOG_INFO, "VSS: %s of %s: %s\n", fs_op, dir, strerror(errno)); + close(fd); + return !!ret; +} + +static int vss_operate(int operation) +{ + char *fs_op; + char match[] = "/dev/"; + FILE *mounts; + struct mntent *ent; + unsigned int cmd; + int error = 0, root_seen = 0; + + switch (operation) { + case VSS_OP_FREEZE: + cmd = FIFREEZE; + fs_op = "freeze"; + break; + case VSS_OP_THAW: + cmd = FITHAW; + fs_op = "thaw"; + break; + default: + return -1; + } + + mounts = setmntent("/proc/mounts", "r"); + if (mounts == NULL) + return -1; + + while ((ent = getmntent(mounts))) { + if (strncmp(ent->mnt_fsname, match, strlen(match))) + continue; + if (strcmp(ent->mnt_type, "iso9660") == 0) + continue; + if (strcmp(ent->mnt_dir, "/") == 0) { + root_seen = 1; + continue; + } + error |= vss_do_freeze(ent->mnt_dir, cmd, fs_op); + } + endmntent(mounts); + + if (root_seen) { + error |= vss_do_freeze("/", cmd, fs_op); + } + + return error; +} + +static int netlink_send(int fd, struct cn_msg *msg) +{ + struct nlmsghdr *nlh; + unsigned int size; + struct msghdr message; + char buffer[64]; + struct iovec iov[2]; + + size = NLMSG_SPACE(sizeof(struct cn_msg) + msg->len); + + nlh = (struct nlmsghdr *)buffer; + nlh->nlmsg_seq = 0; + nlh->nlmsg_pid = getpid(); + nlh->nlmsg_type = NLMSG_DONE; + nlh->nlmsg_len = NLMSG_LENGTH(size - sizeof(*nlh)); + nlh->nlmsg_flags = 0; + + iov[0].iov_base = nlh; + iov[0].iov_len = sizeof(*nlh); + + iov[1].iov_base = msg; + iov[1].iov_len = size; + + memset(&message, 0, sizeof(message)); + message.msg_name = &addr; + message.msg_namelen = sizeof(addr); + message.msg_iov = iov; + message.msg_iovlen = 2; + + return sendmsg(fd, &message, 0); +} + +int main(void) +{ + int fd, len, nl_group; + int error; + struct cn_msg *message; + struct pollfd pfd; + struct nlmsghdr *incoming_msg; + struct cn_msg *incoming_cn_msg; + int op; + struct hv_vss_msg *vss_msg; + + if (daemon(1, 0)) + return 1; + + openlog("Hyper-V VSS", 0, LOG_USER); + syslog(LOG_INFO, "VSS starting; pid is:%d", getpid()); + + fd = socket(AF_NETLINK, SOCK_DGRAM, NETLINK_CONNECTOR); + if (fd < 0) { + syslog(LOG_ERR, "netlink socket creation failed; error:%d", fd); + exit(EXIT_FAILURE); + } + addr.nl_family = AF_NETLINK; + addr.nl_pad = 0; + addr.nl_pid = 0; + addr.nl_groups = 0; + + + error = bind(fd, (struct sockaddr *)&addr, sizeof(addr)); + if (error < 0) { + syslog(LOG_ERR, "bind failed; error:%d", error); + close(fd); + exit(EXIT_FAILURE); + } + nl_group = CN_VSS_IDX; + setsockopt(fd, SOL_NETLINK, NETLINK_ADD_MEMBERSHIP, &nl_group, sizeof(nl_group)); + /* + * Register ourselves with the kernel. + */ + message = (struct cn_msg *)vss_send_buffer; + message->id.idx = CN_VSS_IDX; + message->id.val = CN_VSS_VAL; + message->ack = 0; + vss_msg = (struct hv_vss_msg *)message->data; + vss_msg->vss_hdr.operation = VSS_OP_REGISTER; + + message->len = sizeof(struct hv_vss_msg); + + len = netlink_send(fd, message); + if (len < 0) { + syslog(LOG_ERR, "netlink_send failed; error:%d", len); + close(fd); + exit(EXIT_FAILURE); + } + + pfd.fd = fd; + + while (1) { + struct sockaddr *addr_p = (struct sockaddr *) &addr; + socklen_t addr_l = sizeof(addr); + pfd.events = POLLIN; + pfd.revents = 0; + poll(&pfd, 1, -1); + + len = recvfrom(fd, vss_recv_buffer, sizeof(vss_recv_buffer), 0, + addr_p, &addr_l); + + if (len < 0) { + syslog(LOG_ERR, "recvfrom failed; pid:%u error:%d %s", + addr.nl_pid, errno, strerror(errno)); + close(fd); + return -1; + } + + if (addr.nl_pid) { + syslog(LOG_WARNING, + "Received packet from untrusted pid:%u", + addr.nl_pid); + continue; + } + + incoming_msg = (struct nlmsghdr *)vss_recv_buffer; + + if (incoming_msg->nlmsg_type != NLMSG_DONE) + continue; + + incoming_cn_msg = (struct cn_msg *)NLMSG_DATA(incoming_msg); + vss_msg = (struct hv_vss_msg *)incoming_cn_msg->data; + op = vss_msg->vss_hdr.operation; + error = HV_S_OK; + + switch (op) { + case VSS_OP_FREEZE: + case VSS_OP_THAW: + error = vss_operate(op); + if (error) + error = HV_E_FAIL; + break; + default: + syslog(LOG_ERR, "Illegal op:%d\n", op); + } + vss_msg->error = error; + len = netlink_send(fd, incoming_cn_msg); + if (len < 0) { + syslog(LOG_ERR, "net_link send failed; error:%d", len); + exit(EXIT_FAILURE); + } + } + +} diff --git a/tools/include/tools/be_byteshift.h b/tools/include/tools/be_byteshift.h new file mode 100644 index 000000000..f4912e266 --- /dev/null +++ b/tools/include/tools/be_byteshift.h @@ -0,0 +1,70 @@ +#ifndef _TOOLS_BE_BYTESHIFT_H +#define _TOOLS_BE_BYTESHIFT_H + +#include + +static inline __u16 __get_unaligned_be16(const __u8 *p) +{ + return p[0] << 8 | p[1]; +} + +static inline __u32 __get_unaligned_be32(const __u8 *p) +{ + return p[0] << 24 | p[1] << 16 | p[2] << 8 | p[3]; +} + +static inline __u64 __get_unaligned_be64(const __u8 *p) +{ + return (__u64)__get_unaligned_be32(p) << 32 | + __get_unaligned_be32(p + 4); +} + +static inline void __put_unaligned_be16(__u16 val, __u8 *p) +{ + *p++ = val >> 8; + *p++ = val; +} + +static inline void __put_unaligned_be32(__u32 val, __u8 *p) +{ + __put_unaligned_be16(val >> 16, p); + __put_unaligned_be16(val, p + 2); +} + +static inline void __put_unaligned_be64(__u64 val, __u8 *p) +{ + __put_unaligned_be32(val >> 32, p); + __put_unaligned_be32(val, p + 4); +} + +static inline __u16 get_unaligned_be16(const void *p) +{ + return __get_unaligned_be16((const __u8 *)p); +} + +static inline __u32 get_unaligned_be32(const void *p) +{ + return __get_unaligned_be32((const __u8 *)p); +} + +static inline __u64 get_unaligned_be64(const void *p) +{ + return __get_unaligned_be64((const __u8 *)p); +} + +static inline void put_unaligned_be16(__u16 val, void *p) +{ + __put_unaligned_be16(val, p); +} + +static inline void put_unaligned_be32(__u32 val, void *p) +{ + __put_unaligned_be32(val, p); +} + +static inline void put_unaligned_be64(__u64 val, void *p) +{ + __put_unaligned_be64(val, p); +} + +#endif /* _TOOLS_BE_BYTESHIFT_H */ diff --git a/tools/include/tools/le_byteshift.h b/tools/include/tools/le_byteshift.h new file mode 100644 index 000000000..c99d45a68 --- /dev/null +++ b/tools/include/tools/le_byteshift.h @@ -0,0 +1,70 @@ +#ifndef _TOOLS_LE_BYTESHIFT_H +#define _TOOLS_LE_BYTESHIFT_H + +#include + +static inline __u16 __get_unaligned_le16(const __u8 *p) +{ + return p[0] | p[1] << 8; +} + +static inline __u32 __get_unaligned_le32(const __u8 *p) +{ + return p[0] | p[1] << 8 | p[2] << 16 | p[3] << 24; +} + +static inline __u64 __get_unaligned_le64(const __u8 *p) +{ + return (__u64)__get_unaligned_le32(p + 4) << 32 | + __get_unaligned_le32(p); +} + +static inline void __put_unaligned_le16(__u16 val, __u8 *p) +{ + *p++ = val; + *p++ = val >> 8; +} + +static inline void __put_unaligned_le32(__u32 val, __u8 *p) +{ + __put_unaligned_le16(val >> 16, p + 2); + __put_unaligned_le16(val, p); +} + +static inline void __put_unaligned_le64(__u64 val, __u8 *p) +{ + __put_unaligned_le32(val >> 32, p + 4); + __put_unaligned_le32(val, p); +} + +static inline __u16 get_unaligned_le16(const void *p) +{ + return __get_unaligned_le16((const __u8 *)p); +} + +static inline __u32 get_unaligned_le32(const void *p) +{ + return __get_unaligned_le32((const __u8 *)p); +} + +static inline __u64 get_unaligned_le64(const void *p) +{ + return __get_unaligned_le64((const __u8 *)p); +} + +static inline void put_unaligned_le16(__u16 val, void *p) +{ + __put_unaligned_le16(val, p); +} + +static inline void put_unaligned_le32(__u32 val, void *p) +{ + __put_unaligned_le32(val, p); +} + +static inline void put_unaligned_le64(__u64 val, void *p) +{ + __put_unaligned_le64(val, p); +} + +#endif /* _TOOLS_LE_BYTESHIFT_H */ diff --git a/tools/lguest/.gitignore b/tools/lguest/.gitignore new file mode 100644 index 000000000..115587fd5 --- /dev/null +++ b/tools/lguest/.gitignore @@ -0,0 +1 @@ +lguest diff --git a/tools/lguest/Makefile b/tools/lguest/Makefile new file mode 100644 index 000000000..0ac34206f --- /dev/null +++ b/tools/lguest/Makefile @@ -0,0 +1,8 @@ +# This creates the demonstration utility "lguest" which runs a Linux guest. +# Missing headers? Add "-I../../../include -I../../../arch/x86/include" +CFLAGS:=-m32 -Wall -Wmissing-declarations -Wmissing-prototypes -O3 -U_FORTIFY_SOURCE + +all: lguest + +clean: + rm -f lguest diff --git a/tools/lguest/extract b/tools/lguest/extract new file mode 100644 index 000000000..7730bb6e4 --- /dev/null +++ b/tools/lguest/extract @@ -0,0 +1,58 @@ +#! /bin/sh + +set -e + +PREFIX=$1 +shift + +trap 'rm -r $TMPDIR' 0 +TMPDIR=`mktemp -d` + +exec 3>/dev/null +for f; do + while IFS=" +" read -r LINE; do + case "$LINE" in + *$PREFIX:[0-9]*:\**) + NUM=`echo "$LINE" | sed "s/.*$PREFIX:\([0-9]*\).*/\1/"` + if [ -f $TMPDIR/$NUM ]; then + echo "$TMPDIR/$NUM already exits prior to $f" + exit 1 + fi + exec 3>>$TMPDIR/$NUM + echo $f | sed 's,\.\./,,g' > $TMPDIR/.$NUM + /bin/echo "$LINE" | sed -e "s/$PREFIX:[0-9]*//" -e "s/:\*/*/" >&3 + ;; + *$PREFIX:[0-9]*) + NUM=`echo "$LINE" | sed "s/.*$PREFIX:\([0-9]*\).*/\1/"` + if [ -f $TMPDIR/$NUM ]; then + echo "$TMPDIR/$NUM already exits prior to $f" + exit 1 + fi + exec 3>>$TMPDIR/$NUM + echo $f | sed 's,\.\./,,g' > $TMPDIR/.$NUM + /bin/echo "$LINE" | sed "s/$PREFIX:[0-9]*//" >&3 + ;; + *:\**) + /bin/echo "$LINE" | sed -e "s/:\*/*/" -e "s,/\*\*/,," >&3 + echo >&3 + exec 3>/dev/null + ;; + *) + /bin/echo "$LINE" >&3 + ;; + esac + done < $f + echo >&3 + exec 3>/dev/null +done + +LASTFILE="" +for f in $TMPDIR/*; do + if [ "$LASTFILE" != $(cat $TMPDIR/.$(basename $f) ) ]; then + LASTFILE=$(cat $TMPDIR/.$(basename $f) ) + echo "[ $LASTFILE ]" + fi + cat $f +done + diff --git a/tools/lguest/lguest.c b/tools/lguest/lguest.c new file mode 100644 index 000000000..07a03452c --- /dev/null +++ b/tools/lguest/lguest.c @@ -0,0 +1,2052 @@ +/*P:100 + * This is the Launcher code, a simple program which lays out the "physical" + * memory for the new Guest by mapping the kernel image and the virtual + * devices, then opens /dev/lguest to tell the kernel about the Guest and + * control it. +:*/ +#define _LARGEFILE64_SOURCE +#define _GNU_SOURCE +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include "../../include/linux/lguest_launcher.h" +/*L:110 + * We can ignore the 43 include files we need for this program, but I do want + * to draw attention to the use of kernel-style types. + * + * As Linus said, "C is a Spartan language, and so should your naming be." I + * like these abbreviations, so we define them here. Note that u64 is always + * unsigned long long, which works on all Linux systems: this means that we can + * use %llu in printf for any u64. + */ +typedef unsigned long long u64; +typedef uint32_t u32; +typedef uint16_t u16; +typedef uint8_t u8; +/*:*/ + +#define BRIDGE_PFX "bridge:" +#ifndef SIOCBRADDIF +#define SIOCBRADDIF 0x89a2 /* add interface to bridge */ +#endif +/* We can have up to 256 pages for devices. */ +#define DEVICE_PAGES 256 +/* This will occupy 3 pages: it must be a power of 2. */ +#define VIRTQUEUE_NUM 256 + +/*L:120 + * verbose is both a global flag and a macro. The C preprocessor allows + * this, and although I wouldn't recommend it, it works quite nicely here. + */ +static bool verbose; +#define verbose(args...) \ + do { if (verbose) printf(args); } while(0) +/*:*/ + +/* The pointer to the start of guest memory. */ +static void *guest_base; +/* The maximum guest physical address allowed, and maximum possible. */ +static unsigned long guest_limit, guest_max; +/* The /dev/lguest file descriptor. */ +static int lguest_fd; + +/* a per-cpu variable indicating whose vcpu is currently running */ +static unsigned int __thread cpu_id; + +/* This is our list of devices. */ +struct device_list { + /* Counter to assign interrupt numbers. */ + unsigned int next_irq; + + /* Counter to print out convenient device numbers. */ + unsigned int device_num; + + /* The descriptor page for the devices. */ + u8 *descpage; + + /* A single linked list of devices. */ + struct device *dev; + /* And a pointer to the last device for easy append. */ + struct device *lastdev; +}; + +/* The list of Guest devices, based on command line arguments. */ +static struct device_list devices; + +/* The device structure describes a single device. */ +struct device { + /* The linked-list pointer. */ + struct device *next; + + /* The device's descriptor, as mapped into the Guest. */ + struct lguest_device_desc *desc; + + /* We can't trust desc values once Guest has booted: we use these. */ + unsigned int feature_len; + unsigned int num_vq; + + /* The name of this device, for --verbose. */ + const char *name; + + /* Any queues attached to this device */ + struct virtqueue *vq; + + /* Is it operational */ + bool running; + + /* Device-specific data. */ + void *priv; +}; + +/* The virtqueue structure describes a queue attached to a device. */ +struct virtqueue { + struct virtqueue *next; + + /* Which device owns me. */ + struct device *dev; + + /* The configuration for this queue. */ + struct lguest_vqconfig config; + + /* The actual ring of buffers. */ + struct vring vring; + + /* Last available index we saw. */ + u16 last_avail_idx; + + /* How many are used since we sent last irq? */ + unsigned int pending_used; + + /* Eventfd where Guest notifications arrive. */ + int eventfd; + + /* Function for the thread which is servicing this virtqueue. */ + void (*service)(struct virtqueue *vq); + pid_t thread; +}; + +/* Remember the arguments to the program so we can "reboot" */ +static char **main_args; + +/* The original tty settings to restore on exit. */ +static struct termios orig_term; + +/* + * We have to be careful with barriers: our devices are all run in separate + * threads and so we need to make sure that changes visible to the Guest happen + * in precise order. + */ +#define wmb() __asm__ __volatile__("" : : : "memory") +#define mb() __asm__ __volatile__("" : : : "memory") + +/* Wrapper for the last available index. Makes it easier to change. */ +#define lg_last_avail(vq) ((vq)->last_avail_idx) + +/* + * The virtio configuration space is defined to be little-endian. x86 is + * little-endian too, but it's nice to be explicit so we have these helpers. + */ +#define cpu_to_le16(v16) (v16) +#define cpu_to_le32(v32) (v32) +#define cpu_to_le64(v64) (v64) +#define le16_to_cpu(v16) (v16) +#define le32_to_cpu(v32) (v32) +#define le64_to_cpu(v64) (v64) + +/* Is this iovec empty? */ +static bool iov_empty(const struct iovec iov[], unsigned int num_iov) +{ + unsigned int i; + + for (i = 0; i < num_iov; i++) + if (iov[i].iov_len) + return false; + return true; +} + +/* Take len bytes from the front of this iovec. */ +static void iov_consume(struct iovec iov[], unsigned num_iov, + void *dest, unsigned len) +{ + unsigned int i; + + for (i = 0; i < num_iov; i++) { + unsigned int used; + + used = iov[i].iov_len < len ? iov[i].iov_len : len; + if (dest) { + memcpy(dest, iov[i].iov_base, used); + dest += used; + } + iov[i].iov_base += used; + iov[i].iov_len -= used; + len -= used; + } + if (len != 0) + errx(1, "iovec too short!"); +} + +/* The device virtqueue descriptors are followed by feature bitmasks. */ +static u8 *get_feature_bits(struct device *dev) +{ + return (u8 *)(dev->desc + 1) + + dev->num_vq * sizeof(struct lguest_vqconfig); +} + +/*L:100 + * The Launcher code itself takes us out into userspace, that scary place where + * pointers run wild and free! Unfortunately, like most userspace programs, + * it's quite boring (which is why everyone likes to hack on the kernel!). + * Perhaps if you make up an Lguest Drinking Game at this point, it will get + * you through this section. Or, maybe not. + * + * The Launcher sets up a big chunk of memory to be the Guest's "physical" + * memory and stores it in "guest_base". In other words, Guest physical == + * Launcher virtual with an offset. + * + * This can be tough to get your head around, but usually it just means that we + * use these trivial conversion functions when the Guest gives us its + * "physical" addresses: + */ +static void *from_guest_phys(unsigned long addr) +{ + return guest_base + addr; +} + +static unsigned long to_guest_phys(const void *addr) +{ + return (addr - guest_base); +} + +/*L:130 + * Loading the Kernel. + * + * We start with couple of simple helper routines. open_or_die() avoids + * error-checking code cluttering the callers: + */ +static int open_or_die(const char *name, int flags) +{ + int fd = open(name, flags); + if (fd < 0) + err(1, "Failed to open %s", name); + return fd; +} + +/* map_zeroed_pages() takes a number of pages. */ +static void *map_zeroed_pages(unsigned int num) +{ + int fd = open_or_die("/dev/zero", O_RDONLY); + void *addr; + + /* + * We use a private mapping (ie. if we write to the page, it will be + * copied). We allocate an extra two pages PROT_NONE to act as guard + * pages against read/write attempts that exceed allocated space. + */ + addr = mmap(NULL, getpagesize() * (num+2), + PROT_NONE, MAP_PRIVATE, fd, 0); + + if (addr == MAP_FAILED) + err(1, "Mmapping %u pages of /dev/zero", num); + + if (mprotect(addr + getpagesize(), getpagesize() * num, + PROT_READ|PROT_WRITE) == -1) + err(1, "mprotect rw %u pages failed", num); + + /* + * One neat mmap feature is that you can close the fd, and it + * stays mapped. + */ + close(fd); + + /* Return address after PROT_NONE page */ + return addr + getpagesize(); +} + +/* Get some more pages for a device. */ +static void *get_pages(unsigned int num) +{ + void *addr = from_guest_phys(guest_limit); + + guest_limit += num * getpagesize(); + if (guest_limit > guest_max) + errx(1, "Not enough memory for devices"); + return addr; +} + +/* + * This routine is used to load the kernel or initrd. It tries mmap, but if + * that fails (Plan 9's kernel file isn't nicely aligned on page boundaries), + * it falls back to reading the memory in. + */ +static void map_at(int fd, void *addr, unsigned long offset, unsigned long len) +{ + ssize_t r; + + /* + * We map writable even though for some segments are marked read-only. + * The kernel really wants to be writable: it patches its own + * instructions. + * + * MAP_PRIVATE means that the page won't be copied until a write is + * done to it. This allows us to share untouched memory between + * Guests. + */ + if (mmap(addr, len, PROT_READ|PROT_WRITE, + MAP_FIXED|MAP_PRIVATE, fd, offset) != MAP_FAILED) + return; + + /* pread does a seek and a read in one shot: saves a few lines. */ + r = pread(fd, addr, len, offset); + if (r != len) + err(1, "Reading offset %lu len %lu gave %zi", offset, len, r); +} + +/* + * This routine takes an open vmlinux image, which is in ELF, and maps it into + * the Guest memory. ELF = Embedded Linking Format, which is the format used + * by all modern binaries on Linux including the kernel. + * + * The ELF headers give *two* addresses: a physical address, and a virtual + * address. We use the physical address; the Guest will map itself to the + * virtual address. + * + * We return the starting address. + */ +static unsigned long map_elf(int elf_fd, const Elf32_Ehdr *ehdr) +{ + Elf32_Phdr phdr[ehdr->e_phnum]; + unsigned int i; + + /* + * Sanity checks on the main ELF header: an x86 executable with a + * reasonable number of correctly-sized program headers. + */ + if (ehdr->e_type != ET_EXEC + || ehdr->e_machine != EM_386 + || ehdr->e_phentsize != sizeof(Elf32_Phdr) + || ehdr->e_phnum < 1 || ehdr->e_phnum > 65536U/sizeof(Elf32_Phdr)) + errx(1, "Malformed elf header"); + + /* + * An ELF executable contains an ELF header and a number of "program" + * headers which indicate which parts ("segments") of the program to + * load where. + */ + + /* We read in all the program headers at once: */ + if (lseek(elf_fd, ehdr->e_phoff, SEEK_SET) < 0) + err(1, "Seeking to program headers"); + if (read(elf_fd, phdr, sizeof(phdr)) != sizeof(phdr)) + err(1, "Reading program headers"); + + /* + * Try all the headers: there are usually only three. A read-only one, + * a read-write one, and a "note" section which we don't load. + */ + for (i = 0; i < ehdr->e_phnum; i++) { + /* If this isn't a loadable segment, we ignore it */ + if (phdr[i].p_type != PT_LOAD) + continue; + + verbose("Section %i: size %i addr %p\n", + i, phdr[i].p_memsz, (void *)phdr[i].p_paddr); + + /* We map this section of the file at its physical address. */ + map_at(elf_fd, from_guest_phys(phdr[i].p_paddr), + phdr[i].p_offset, phdr[i].p_filesz); + } + + /* The entry point is given in the ELF header. */ + return ehdr->e_entry; +} + +/*L:150 + * A bzImage, unlike an ELF file, is not meant to be loaded. You're supposed + * to jump into it and it will unpack itself. We used to have to perform some + * hairy magic because the unpacking code scared me. + * + * Fortunately, Jeremy Fitzhardinge convinced me it wasn't that hard and wrote + * a small patch to jump over the tricky bits in the Guest, so now we just read + * the funky header so we know where in the file to load, and away we go! + */ +static unsigned long load_bzimage(int fd) +{ + struct boot_params boot; + int r; + /* Modern bzImages get loaded at 1M. */ + void *p = from_guest_phys(0x100000); + + /* + * Go back to the start of the file and read the header. It should be + * a Linux boot header (see Documentation/x86/boot.txt) + */ + lseek(fd, 0, SEEK_SET); + read(fd, &boot, sizeof(boot)); + + /* Inside the setup_hdr, we expect the magic "HdrS" */ + if (memcmp(&boot.hdr.header, "HdrS", 4) != 0) + errx(1, "This doesn't look like a bzImage to me"); + + /* Skip over the extra sectors of the header. */ + lseek(fd, (boot.hdr.setup_sects+1) * 512, SEEK_SET); + + /* Now read everything into memory. in nice big chunks. */ + while ((r = read(fd, p, 65536)) > 0) + p += r; + + /* Finally, code32_start tells us where to enter the kernel. */ + return boot.hdr.code32_start; +} + +/*L:140 + * Loading the kernel is easy when it's a "vmlinux", but most kernels + * come wrapped up in the self-decompressing "bzImage" format. With a little + * work, we can load those, too. + */ +static unsigned long load_kernel(int fd) +{ + Elf32_Ehdr hdr; + + /* Read in the first few bytes. */ + if (read(fd, &hdr, sizeof(hdr)) != sizeof(hdr)) + err(1, "Reading kernel"); + + /* If it's an ELF file, it starts with "\177ELF" */ + if (memcmp(hdr.e_ident, ELFMAG, SELFMAG) == 0) + return map_elf(fd, &hdr); + + /* Otherwise we assume it's a bzImage, and try to load it. */ + return load_bzimage(fd); +} + +/* + * This is a trivial little helper to align pages. Andi Kleen hated it because + * it calls getpagesize() twice: "it's dumb code." + * + * Kernel guys get really het up about optimization, even when it's not + * necessary. I leave this code as a reaction against that. + */ +static inline unsigned long page_align(unsigned long addr) +{ + /* Add upwards and truncate downwards. */ + return ((addr + getpagesize()-1) & ~(getpagesize()-1)); +} + +/*L:180 + * An "initial ram disk" is a disk image loaded into memory along with the + * kernel which the kernel can use to boot from without needing any drivers. + * Most distributions now use this as standard: the initrd contains the code to + * load the appropriate driver modules for the current machine. + * + * Importantly, James Morris works for RedHat, and Fedora uses initrds for its + * kernels. He sent me this (and tells me when I break it). + */ +static unsigned long load_initrd(const char *name, unsigned long mem) +{ + int ifd; + struct stat st; + unsigned long len; + + ifd = open_or_die(name, O_RDONLY); + /* fstat() is needed to get the file size. */ + if (fstat(ifd, &st) < 0) + err(1, "fstat() on initrd '%s'", name); + + /* + * We map the initrd at the top of memory, but mmap wants it to be + * page-aligned, so we round the size up for that. + */ + len = page_align(st.st_size); + map_at(ifd, from_guest_phys(mem - len), 0, st.st_size); + /* + * Once a file is mapped, you can close the file descriptor. It's a + * little odd, but quite useful. + */ + close(ifd); + verbose("mapped initrd %s size=%lu @ %p\n", name, len, (void*)mem-len); + + /* We return the initrd size. */ + return len; +} +/*:*/ + +/* + * Simple routine to roll all the commandline arguments together with spaces + * between them. + */ +static void concat(char *dst, char *args[]) +{ + unsigned int i, len = 0; + + for (i = 0; args[i]; i++) { + if (i) { + strcat(dst+len, " "); + len++; + } + strcpy(dst+len, args[i]); + len += strlen(args[i]); + } + /* In case it's empty. */ + dst[len] = '\0'; +} + +/*L:185 + * This is where we actually tell the kernel to initialize the Guest. We + * saw the arguments it expects when we looked at initialize() in lguest_user.c: + * the base of Guest "physical" memory, the top physical page to allow and the + * entry point for the Guest. + */ +static void tell_kernel(unsigned long start) +{ + unsigned long args[] = { LHREQ_INITIALIZE, + (unsigned long)guest_base, + guest_limit / getpagesize(), start }; + verbose("Guest: %p - %p (%#lx)\n", + guest_base, guest_base + guest_limit, guest_limit); + lguest_fd = open_or_die("/dev/lguest", O_RDWR); + if (write(lguest_fd, args, sizeof(args)) < 0) + err(1, "Writing to /dev/lguest"); +} +/*:*/ + +/*L:200 + * Device Handling. + * + * When the Guest gives us a buffer, it sends an array of addresses and sizes. + * We need to make sure it's not trying to reach into the Launcher itself, so + * we have a convenient routine which checks it and exits with an error message + * if something funny is going on: + */ +static void *_check_pointer(unsigned long addr, unsigned int size, + unsigned int line) +{ + /* + * Check if the requested address and size exceeds the allocated memory, + * or addr + size wraps around. + */ + if ((addr + size) > guest_limit || (addr + size) < addr) + errx(1, "%s:%i: Invalid address %#lx", __FILE__, line, addr); + /* + * We return a pointer for the caller's convenience, now we know it's + * safe to use. + */ + return from_guest_phys(addr); +} +/* A macro which transparently hands the line number to the real function. */ +#define check_pointer(addr,size) _check_pointer(addr, size, __LINE__) + +/* + * Each buffer in the virtqueues is actually a chain of descriptors. This + * function returns the next descriptor in the chain, or vq->vring.num if we're + * at the end. + */ +static unsigned next_desc(struct vring_desc *desc, + unsigned int i, unsigned int max) +{ + unsigned int next; + + /* If this descriptor says it doesn't chain, we're done. */ + if (!(desc[i].flags & VRING_DESC_F_NEXT)) + return max; + + /* Check they're not leading us off end of descriptors. */ + next = desc[i].next; + /* Make sure compiler knows to grab that: we don't want it changing! */ + wmb(); + + if (next >= max) + errx(1, "Desc next is %u", next); + + return next; +} + +/* + * This actually sends the interrupt for this virtqueue, if we've used a + * buffer. + */ +static void trigger_irq(struct virtqueue *vq) +{ + unsigned long buf[] = { LHREQ_IRQ, vq->config.irq }; + + /* Don't inform them if nothing used. */ + if (!vq->pending_used) + return; + vq->pending_used = 0; + + /* If they don't want an interrupt, don't send one... */ + if (vq->vring.avail->flags & VRING_AVAIL_F_NO_INTERRUPT) { + return; + } + + /* Send the Guest an interrupt tell them we used something up. */ + if (write(lguest_fd, buf, sizeof(buf)) != 0) + err(1, "Triggering irq %i", vq->config.irq); +} + +/* + * This looks in the virtqueue for the first available buffer, and converts + * it to an iovec for convenient access. Since descriptors consist of some + * number of output then some number of input descriptors, it's actually two + * iovecs, but we pack them into one and note how many of each there were. + * + * This function waits if necessary, and returns the descriptor number found. + */ +static unsigned wait_for_vq_desc(struct virtqueue *vq, + struct iovec iov[], + unsigned int *out_num, unsigned int *in_num) +{ + unsigned int i, head, max; + struct vring_desc *desc; + u16 last_avail = lg_last_avail(vq); + + /* There's nothing available? */ + while (last_avail == vq->vring.avail->idx) { + u64 event; + + /* + * Since we're about to sleep, now is a good time to tell the + * Guest about what we've used up to now. + */ + trigger_irq(vq); + + /* OK, now we need to know about added descriptors. */ + vq->vring.used->flags &= ~VRING_USED_F_NO_NOTIFY; + + /* + * They could have slipped one in as we were doing that: make + * sure it's written, then check again. + */ + mb(); + if (last_avail != vq->vring.avail->idx) { + vq->vring.used->flags |= VRING_USED_F_NO_NOTIFY; + break; + } + + /* Nothing new? Wait for eventfd to tell us they refilled. */ + if (read(vq->eventfd, &event, sizeof(event)) != sizeof(event)) + errx(1, "Event read failed?"); + + /* We don't need to be notified again. */ + vq->vring.used->flags |= VRING_USED_F_NO_NOTIFY; + } + + /* Check it isn't doing very strange things with descriptor numbers. */ + if ((u16)(vq->vring.avail->idx - last_avail) > vq->vring.num) + errx(1, "Guest moved used index from %u to %u", + last_avail, vq->vring.avail->idx); + + /* + * Grab the next descriptor number they're advertising, and increment + * the index we've seen. + */ + head = vq->vring.avail->ring[last_avail % vq->vring.num]; + lg_last_avail(vq)++; + + /* If their number is silly, that's a fatal mistake. */ + if (head >= vq->vring.num) + errx(1, "Guest says index %u is available", head); + + /* When we start there are none of either input nor output. */ + *out_num = *in_num = 0; + + max = vq->vring.num; + desc = vq->vring.desc; + i = head; + + /* + * If this is an indirect entry, then this buffer contains a descriptor + * table which we handle as if it's any normal descriptor chain. + */ + if (desc[i].flags & VRING_DESC_F_INDIRECT) { + if (desc[i].len % sizeof(struct vring_desc)) + errx(1, "Invalid size for indirect buffer table"); + + max = desc[i].len / sizeof(struct vring_desc); + desc = check_pointer(desc[i].addr, desc[i].len); + i = 0; + } + + do { + /* Grab the first descriptor, and check it's OK. */ + iov[*out_num + *in_num].iov_len = desc[i].len; + iov[*out_num + *in_num].iov_base + = check_pointer(desc[i].addr, desc[i].len); + /* If this is an input descriptor, increment that count. */ + if (desc[i].flags & VRING_DESC_F_WRITE) + (*in_num)++; + else { + /* + * If it's an output descriptor, they're all supposed + * to come before any input descriptors. + */ + if (*in_num) + errx(1, "Descriptor has out after in"); + (*out_num)++; + } + + /* If we've got too many, that implies a descriptor loop. */ + if (*out_num + *in_num > max) + errx(1, "Looped descriptor"); + } while ((i = next_desc(desc, i, max)) != max); + + return head; +} + +/* + * After we've used one of their buffers, we tell the Guest about it. Sometime + * later we'll want to send them an interrupt using trigger_irq(); note that + * wait_for_vq_desc() does that for us if it has to wait. + */ +static void add_used(struct virtqueue *vq, unsigned int head, int len) +{ + struct vring_used_elem *used; + + /* + * The virtqueue contains a ring of used buffers. Get a pointer to the + * next entry in that used ring. + */ + used = &vq->vring.used->ring[vq->vring.used->idx % vq->vring.num]; + used->id = head; + used->len = len; + /* Make sure buffer is written before we update index. */ + wmb(); + vq->vring.used->idx++; + vq->pending_used++; +} + +/* And here's the combo meal deal. Supersize me! */ +static void add_used_and_trigger(struct virtqueue *vq, unsigned head, int len) +{ + add_used(vq, head, len); + trigger_irq(vq); +} + +/* + * The Console + * + * We associate some data with the console for our exit hack. + */ +struct console_abort { + /* How many times have they hit ^C? */ + int count; + /* When did they start? */ + struct timeval start; +}; + +/* This is the routine which handles console input (ie. stdin). */ +static void console_input(struct virtqueue *vq) +{ + int len; + unsigned int head, in_num, out_num; + struct console_abort *abort = vq->dev->priv; + struct iovec iov[vq->vring.num]; + + /* Make sure there's a descriptor available. */ + head = wait_for_vq_desc(vq, iov, &out_num, &in_num); + if (out_num) + errx(1, "Output buffers in console in queue?"); + + /* Read into it. This is where we usually wait. */ + len = readv(STDIN_FILENO, iov, in_num); + if (len <= 0) { + /* Ran out of input? */ + warnx("Failed to get console input, ignoring console."); + /* + * For simplicity, dying threads kill the whole Launcher. So + * just nap here. + */ + for (;;) + pause(); + } + + /* Tell the Guest we used a buffer. */ + add_used_and_trigger(vq, head, len); + + /* + * Three ^C within one second? Exit. + * + * This is such a hack, but works surprisingly well. Each ^C has to + * be in a buffer by itself, so they can't be too fast. But we check + * that we get three within about a second, so they can't be too + * slow. + */ + if (len != 1 || ((char *)iov[0].iov_base)[0] != 3) { + abort->count = 0; + return; + } + + abort->count++; + if (abort->count == 1) + gettimeofday(&abort->start, NULL); + else if (abort->count == 3) { + struct timeval now; + gettimeofday(&now, NULL); + /* Kill all Launcher processes with SIGINT, like normal ^C */ + if (now.tv_sec <= abort->start.tv_sec+1) + kill(0, SIGINT); + abort->count = 0; + } +} + +/* This is the routine which handles console output (ie. stdout). */ +static void console_output(struct virtqueue *vq) +{ + unsigned int head, out, in; + struct iovec iov[vq->vring.num]; + + /* We usually wait in here, for the Guest to give us something. */ + head = wait_for_vq_desc(vq, iov, &out, &in); + if (in) + errx(1, "Input buffers in console output queue?"); + + /* writev can return a partial write, so we loop here. */ + while (!iov_empty(iov, out)) { + int len = writev(STDOUT_FILENO, iov, out); + if (len <= 0) { + warn("Write to stdout gave %i (%d)", len, errno); + break; + } + iov_consume(iov, out, NULL, len); + } + + /* + * We're finished with that buffer: if we're going to sleep, + * wait_for_vq_desc() will prod the Guest with an interrupt. + */ + add_used(vq, head, 0); +} + +/* + * The Network + * + * Handling output for network is also simple: we get all the output buffers + * and write them to /dev/net/tun. + */ +struct net_info { + int tunfd; +}; + +static void net_output(struct virtqueue *vq) +{ + struct net_info *net_info = vq->dev->priv; + unsigned int head, out, in; + struct iovec iov[vq->vring.num]; + + /* We usually wait in here for the Guest to give us a packet. */ + head = wait_for_vq_desc(vq, iov, &out, &in); + if (in) + errx(1, "Input buffers in net output queue?"); + /* + * Send the whole thing through to /dev/net/tun. It expects the exact + * same format: what a coincidence! + */ + if (writev(net_info->tunfd, iov, out) < 0) + warnx("Write to tun failed (%d)?", errno); + + /* + * Done with that one; wait_for_vq_desc() will send the interrupt if + * all packets are processed. + */ + add_used(vq, head, 0); +} + +/* + * Handling network input is a bit trickier, because I've tried to optimize it. + * + * First we have a helper routine which tells is if from this file descriptor + * (ie. the /dev/net/tun device) will block: + */ +static bool will_block(int fd) +{ + fd_set fdset; + struct timeval zero = { 0, 0 }; + FD_ZERO(&fdset); + FD_SET(fd, &fdset); + return select(fd+1, &fdset, NULL, NULL, &zero) != 1; +} + +/* + * This handles packets coming in from the tun device to our Guest. Like all + * service routines, it gets called again as soon as it returns, so you don't + * see a while(1) loop here. + */ +static void net_input(struct virtqueue *vq) +{ + int len; + unsigned int head, out, in; + struct iovec iov[vq->vring.num]; + struct net_info *net_info = vq->dev->priv; + + /* + * Get a descriptor to write an incoming packet into. This will also + * send an interrupt if they're out of descriptors. + */ + head = wait_for_vq_desc(vq, iov, &out, &in); + if (out) + errx(1, "Output buffers in net input queue?"); + + /* + * If it looks like we'll block reading from the tun device, send them + * an interrupt. + */ + if (vq->pending_used && will_block(net_info->tunfd)) + trigger_irq(vq); + + /* + * Read in the packet. This is where we normally wait (when there's no + * incoming network traffic). + */ + len = readv(net_info->tunfd, iov, in); + if (len <= 0) + warn("Failed to read from tun (%d).", errno); + + /* + * Mark that packet buffer as used, but don't interrupt here. We want + * to wait until we've done as much work as we can. + */ + add_used(vq, head, len); +} +/*:*/ + +/* This is the helper to create threads: run the service routine in a loop. */ +static int do_thread(void *_vq) +{ + struct virtqueue *vq = _vq; + + for (;;) + vq->service(vq); + return 0; +} + +/* + * When a child dies, we kill our entire process group with SIGTERM. This + * also has the side effect that the shell restores the console for us! + */ +static void kill_launcher(int signal) +{ + kill(0, SIGTERM); +} + +static void reset_device(struct device *dev) +{ + struct virtqueue *vq; + + verbose("Resetting device %s\n", dev->name); + + /* Clear any features they've acked. */ + memset(get_feature_bits(dev) + dev->feature_len, 0, dev->feature_len); + + /* We're going to be explicitly killing threads, so ignore them. */ + signal(SIGCHLD, SIG_IGN); + + /* Zero out the virtqueues, get rid of their threads */ + for (vq = dev->vq; vq; vq = vq->next) { + if (vq->thread != (pid_t)-1) { + kill(vq->thread, SIGTERM); + waitpid(vq->thread, NULL, 0); + vq->thread = (pid_t)-1; + } + memset(vq->vring.desc, 0, + vring_size(vq->config.num, LGUEST_VRING_ALIGN)); + lg_last_avail(vq) = 0; + } + dev->running = false; + + /* Now we care if threads die. */ + signal(SIGCHLD, (void *)kill_launcher); +} + +/*L:216 + * This actually creates the thread which services the virtqueue for a device. + */ +static void create_thread(struct virtqueue *vq) +{ + /* + * Create stack for thread. Since the stack grows upwards, we point + * the stack pointer to the end of this region. + */ + char *stack = malloc(32768); + unsigned long args[] = { LHREQ_EVENTFD, + vq->config.pfn*getpagesize(), 0 }; + + /* Create a zero-initialized eventfd. */ + vq->eventfd = eventfd(0, 0); + if (vq->eventfd < 0) + err(1, "Creating eventfd"); + args[2] = vq->eventfd; + + /* + * Attach an eventfd to this virtqueue: it will go off when the Guest + * does an LHCALL_NOTIFY for this vq. + */ + if (write(lguest_fd, &args, sizeof(args)) != 0) + err(1, "Attaching eventfd"); + + /* + * CLONE_VM: because it has to access the Guest memory, and SIGCHLD so + * we get a signal if it dies. + */ + vq->thread = clone(do_thread, stack + 32768, CLONE_VM | SIGCHLD, vq); + if (vq->thread == (pid_t)-1) + err(1, "Creating clone"); + + /* We close our local copy now the child has it. */ + close(vq->eventfd); +} + +static void start_device(struct device *dev) +{ + unsigned int i; + struct virtqueue *vq; + + verbose("Device %s OK: offered", dev->name); + for (i = 0; i < dev->feature_len; i++) + verbose(" %02x", get_feature_bits(dev)[i]); + verbose(", accepted"); + for (i = 0; i < dev->feature_len; i++) + verbose(" %02x", get_feature_bits(dev) + [dev->feature_len+i]); + + for (vq = dev->vq; vq; vq = vq->next) { + if (vq->service) + create_thread(vq); + } + dev->running = true; +} + +static void cleanup_devices(void) +{ + struct device *dev; + + for (dev = devices.dev; dev; dev = dev->next) + reset_device(dev); + + /* If we saved off the original terminal settings, restore them now. */ + if (orig_term.c_lflag & (ISIG|ICANON|ECHO)) + tcsetattr(STDIN_FILENO, TCSANOW, &orig_term); +} + +/* When the Guest tells us they updated the status field, we handle it. */ +static void update_device_status(struct device *dev) +{ + /* A zero status is a reset, otherwise it's a set of flags. */ + if (dev->desc->status == 0) + reset_device(dev); + else if (dev->desc->status & VIRTIO_CONFIG_S_FAILED) { + warnx("Device %s configuration FAILED", dev->name); + if (dev->running) + reset_device(dev); + } else { + if (dev->running) + err(1, "Device %s features finalized twice", dev->name); + start_device(dev); + } +} + +/*L:215 + * This is the generic routine we call when the Guest uses LHCALL_NOTIFY. In + * particular, it's used to notify us of device status changes during boot. + */ +static void handle_output(unsigned long addr) +{ + struct device *i; + + /* Check each device. */ + for (i = devices.dev; i; i = i->next) { + struct virtqueue *vq; + + /* + * Notifications to device descriptors mean they updated the + * device status. + */ + if (from_guest_phys(addr) == i->desc) { + update_device_status(i); + return; + } + + /* Devices should not be used before features are finalized. */ + for (vq = i->vq; vq; vq = vq->next) { + if (addr != vq->config.pfn*getpagesize()) + continue; + errx(1, "Notification on %s before setup!", i->name); + } + } + + /* + * Early console write is done using notify on a nul-terminated string + * in Guest memory. It's also great for hacking debugging messages + * into a Guest. + */ + if (addr >= guest_limit) + errx(1, "Bad NOTIFY %#lx", addr); + + write(STDOUT_FILENO, from_guest_phys(addr), + strnlen(from_guest_phys(addr), guest_limit - addr)); +} + +/*L:190 + * Device Setup + * + * All devices need a descriptor so the Guest knows it exists, and a "struct + * device" so the Launcher can keep track of it. We have common helper + * routines to allocate and manage them. + */ + +/* + * The layout of the device page is a "struct lguest_device_desc" followed by a + * number of virtqueue descriptors, then two sets of feature bits, then an + * array of configuration bytes. This routine returns the configuration + * pointer. + */ +static u8 *device_config(const struct device *dev) +{ + return (void *)(dev->desc + 1) + + dev->num_vq * sizeof(struct lguest_vqconfig) + + dev->feature_len * 2; +} + +/* + * This routine allocates a new "struct lguest_device_desc" from descriptor + * table page just above the Guest's normal memory. It returns a pointer to + * that descriptor. + */ +static struct lguest_device_desc *new_dev_desc(u16 type) +{ + struct lguest_device_desc d = { .type = type }; + void *p; + + /* Figure out where the next device config is, based on the last one. */ + if (devices.lastdev) + p = device_config(devices.lastdev) + + devices.lastdev->desc->config_len; + else + p = devices.descpage; + + /* We only have one page for all the descriptors. */ + if (p + sizeof(d) > (void *)devices.descpage + getpagesize()) + errx(1, "Too many devices"); + + /* p might not be aligned, so we memcpy in. */ + return memcpy(p, &d, sizeof(d)); +} + +/* + * Each device descriptor is followed by the description of its virtqueues. We + * specify how many descriptors the virtqueue is to have. + */ +static void add_virtqueue(struct device *dev, unsigned int num_descs, + void (*service)(struct virtqueue *)) +{ + unsigned int pages; + struct virtqueue **i, *vq = malloc(sizeof(*vq)); + void *p; + + /* First we need some memory for this virtqueue. */ + pages = (vring_size(num_descs, LGUEST_VRING_ALIGN) + getpagesize() - 1) + / getpagesize(); + p = get_pages(pages); + + /* Initialize the virtqueue */ + vq->next = NULL; + vq->last_avail_idx = 0; + vq->dev = dev; + + /* + * This is the routine the service thread will run, and its Process ID + * once it's running. + */ + vq->service = service; + vq->thread = (pid_t)-1; + + /* Initialize the configuration. */ + vq->config.num = num_descs; + vq->config.irq = devices.next_irq++; + vq->config.pfn = to_guest_phys(p) / getpagesize(); + + /* Initialize the vring. */ + vring_init(&vq->vring, num_descs, p, LGUEST_VRING_ALIGN); + + /* + * Append virtqueue to this device's descriptor. We use + * device_config() to get the end of the device's current virtqueues; + * we check that we haven't added any config or feature information + * yet, otherwise we'd be overwriting them. + */ + assert(dev->desc->config_len == 0 && dev->desc->feature_len == 0); + memcpy(device_config(dev), &vq->config, sizeof(vq->config)); + dev->num_vq++; + dev->desc->num_vq++; + + verbose("Virtqueue page %#lx\n", to_guest_phys(p)); + + /* + * Add to tail of list, so dev->vq is first vq, dev->vq->next is + * second. + */ + for (i = &dev->vq; *i; i = &(*i)->next); + *i = vq; +} + +/* + * The first half of the feature bitmask is for us to advertise features. The + * second half is for the Guest to accept features. + */ +static void add_feature(struct device *dev, unsigned bit) +{ + u8 *features = get_feature_bits(dev); + + /* We can't extend the feature bits once we've added config bytes */ + if (dev->desc->feature_len <= bit / CHAR_BIT) { + assert(dev->desc->config_len == 0); + dev->feature_len = dev->desc->feature_len = (bit/CHAR_BIT) + 1; + } + + features[bit / CHAR_BIT] |= (1 << (bit % CHAR_BIT)); +} + +/* + * This routine sets the configuration fields for an existing device's + * descriptor. It only works for the last device, but that's OK because that's + * how we use it. + */ +static void set_config(struct device *dev, unsigned len, const void *conf) +{ + /* Check we haven't overflowed our single page. */ + if (device_config(dev) + len > devices.descpage + getpagesize()) + errx(1, "Too many devices"); + + /* Copy in the config information, and store the length. */ + memcpy(device_config(dev), conf, len); + dev->desc->config_len = len; + + /* Size must fit in config_len field (8 bits)! */ + assert(dev->desc->config_len == len); +} + +/* + * This routine does all the creation and setup of a new device, including + * calling new_dev_desc() to allocate the descriptor and device memory. We + * don't actually start the service threads until later. + * + * See what I mean about userspace being boring? + */ +static struct device *new_device(const char *name, u16 type) +{ + struct device *dev = malloc(sizeof(*dev)); + + /* Now we populate the fields one at a time. */ + dev->desc = new_dev_desc(type); + dev->name = name; + dev->vq = NULL; + dev->feature_len = 0; + dev->num_vq = 0; + dev->running = false; + dev->next = NULL; + + /* + * Append to device list. Prepending to a single-linked list is + * easier, but the user expects the devices to be arranged on the bus + * in command-line order. The first network device on the command line + * is eth0, the first block device /dev/vda, etc. + */ + if (devices.lastdev) + devices.lastdev->next = dev; + else + devices.dev = dev; + devices.lastdev = dev; + + return dev; +} + +/* + * Our first setup routine is the console. It's a fairly simple device, but + * UNIX tty handling makes it uglier than it could be. + */ +static void setup_console(void) +{ + struct device *dev; + + /* If we can save the initial standard input settings... */ + if (tcgetattr(STDIN_FILENO, &orig_term) == 0) { + struct termios term = orig_term; + /* + * Then we turn off echo, line buffering and ^C etc: We want a + * raw input stream to the Guest. + */ + term.c_lflag &= ~(ISIG|ICANON|ECHO); + tcsetattr(STDIN_FILENO, TCSANOW, &term); + } + + dev = new_device("console", VIRTIO_ID_CONSOLE); + + /* We store the console state in dev->priv, and initialize it. */ + dev->priv = malloc(sizeof(struct console_abort)); + ((struct console_abort *)dev->priv)->count = 0; + + /* + * The console needs two virtqueues: the input then the output. When + * they put something the input queue, we make sure we're listening to + * stdin. When they put something in the output queue, we write it to + * stdout. + */ + add_virtqueue(dev, VIRTQUEUE_NUM, console_input); + add_virtqueue(dev, VIRTQUEUE_NUM, console_output); + + verbose("device %u: console\n", ++devices.device_num); +} +/*:*/ + +/*M:010 + * Inter-guest networking is an interesting area. Simplest is to have a + * --sharenet= option which opens or creates a named pipe. This can be + * used to send packets to another guest in a 1:1 manner. + * + * More sophisticated is to use one of the tools developed for project like UML + * to do networking. + * + * Faster is to do virtio bonding in kernel. Doing this 1:1 would be + * completely generic ("here's my vring, attach to your vring") and would work + * for any traffic. Of course, namespace and permissions issues need to be + * dealt with. A more sophisticated "multi-channel" virtio_net.c could hide + * multiple inter-guest channels behind one interface, although it would + * require some manner of hotplugging new virtio channels. + * + * Finally, we could use a virtio network switch in the kernel, ie. vhost. +:*/ + +static u32 str2ip(const char *ipaddr) +{ + unsigned int b[4]; + + if (sscanf(ipaddr, "%u.%u.%u.%u", &b[0], &b[1], &b[2], &b[3]) != 4) + errx(1, "Failed to parse IP address '%s'", ipaddr); + return (b[0] << 24) | (b[1] << 16) | (b[2] << 8) | b[3]; +} + +static void str2mac(const char *macaddr, unsigned char mac[6]) +{ + unsigned int m[6]; + if (sscanf(macaddr, "%02x:%02x:%02x:%02x:%02x:%02x", + &m[0], &m[1], &m[2], &m[3], &m[4], &m[5]) != 6) + errx(1, "Failed to parse mac address '%s'", macaddr); + mac[0] = m[0]; + mac[1] = m[1]; + mac[2] = m[2]; + mac[3] = m[3]; + mac[4] = m[4]; + mac[5] = m[5]; +} + +/* + * This code is "adapted" from libbridge: it attaches the Host end of the + * network device to the bridge device specified by the command line. + * + * This is yet another James Morris contribution (I'm an IP-level guy, so I + * dislike bridging), and I just try not to break it. + */ +static void add_to_bridge(int fd, const char *if_name, const char *br_name) +{ + int ifidx; + struct ifreq ifr; + + if (!*br_name) + errx(1, "must specify bridge name"); + + ifidx = if_nametoindex(if_name); + if (!ifidx) + errx(1, "interface %s does not exist!", if_name); + + strncpy(ifr.ifr_name, br_name, IFNAMSIZ); + ifr.ifr_name[IFNAMSIZ-1] = '\0'; + ifr.ifr_ifindex = ifidx; + if (ioctl(fd, SIOCBRADDIF, &ifr) < 0) + err(1, "can't add %s to bridge %s", if_name, br_name); +} + +/* + * This sets up the Host end of the network device with an IP address, brings + * it up so packets will flow, the copies the MAC address into the hwaddr + * pointer. + */ +static void configure_device(int fd, const char *tapif, u32 ipaddr) +{ + struct ifreq ifr; + struct sockaddr_in sin; + + memset(&ifr, 0, sizeof(ifr)); + strcpy(ifr.ifr_name, tapif); + + /* Don't read these incantations. Just cut & paste them like I did! */ + sin.sin_family = AF_INET; + sin.sin_addr.s_addr = htonl(ipaddr); + memcpy(&ifr.ifr_addr, &sin, sizeof(sin)); + if (ioctl(fd, SIOCSIFADDR, &ifr) != 0) + err(1, "Setting %s interface address", tapif); + ifr.ifr_flags = IFF_UP; + if (ioctl(fd, SIOCSIFFLAGS, &ifr) != 0) + err(1, "Bringing interface %s up", tapif); +} + +static int get_tun_device(char tapif[IFNAMSIZ]) +{ + struct ifreq ifr; + int netfd; + + /* Start with this zeroed. Messy but sure. */ + memset(&ifr, 0, sizeof(ifr)); + + /* + * We open the /dev/net/tun device and tell it we want a tap device. A + * tap device is like a tun device, only somehow different. To tell + * the truth, I completely blundered my way through this code, but it + * works now! + */ + netfd = open_or_die("/dev/net/tun", O_RDWR); + ifr.ifr_flags = IFF_TAP | IFF_NO_PI | IFF_VNET_HDR; + strcpy(ifr.ifr_name, "tap%d"); + if (ioctl(netfd, TUNSETIFF, &ifr) != 0) + err(1, "configuring /dev/net/tun"); + + if (ioctl(netfd, TUNSETOFFLOAD, + TUN_F_CSUM|TUN_F_TSO4|TUN_F_TSO6|TUN_F_TSO_ECN) != 0) + err(1, "Could not set features for tun device"); + + /* + * We don't need checksums calculated for packets coming in this + * device: trust us! + */ + ioctl(netfd, TUNSETNOCSUM, 1); + + memcpy(tapif, ifr.ifr_name, IFNAMSIZ); + return netfd; +} + +/*L:195 + * Our network is a Host<->Guest network. This can either use bridging or + * routing, but the principle is the same: it uses the "tun" device to inject + * packets into the Host as if they came in from a normal network card. We + * just shunt packets between the Guest and the tun device. + */ +static void setup_tun_net(char *arg) +{ + struct device *dev; + struct net_info *net_info = malloc(sizeof(*net_info)); + int ipfd; + u32 ip = INADDR_ANY; + bool bridging = false; + char tapif[IFNAMSIZ], *p; + struct virtio_net_config conf; + + net_info->tunfd = get_tun_device(tapif); + + /* First we create a new network device. */ + dev = new_device("net", VIRTIO_ID_NET); + dev->priv = net_info; + + /* Network devices need a recv and a send queue, just like console. */ + add_virtqueue(dev, VIRTQUEUE_NUM, net_input); + add_virtqueue(dev, VIRTQUEUE_NUM, net_output); + + /* + * We need a socket to perform the magic network ioctls to bring up the + * tap interface, connect to the bridge etc. Any socket will do! + */ + ipfd = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP); + if (ipfd < 0) + err(1, "opening IP socket"); + + /* If the command line was --tunnet=bridge: do bridging. */ + if (!strncmp(BRIDGE_PFX, arg, strlen(BRIDGE_PFX))) { + arg += strlen(BRIDGE_PFX); + bridging = true; + } + + /* A mac address may follow the bridge name or IP address */ + p = strchr(arg, ':'); + if (p) { + str2mac(p+1, conf.mac); + add_feature(dev, VIRTIO_NET_F_MAC); + *p = '\0'; + } + + /* arg is now either an IP address or a bridge name */ + if (bridging) + add_to_bridge(ipfd, tapif, arg); + else + ip = str2ip(arg); + + /* Set up the tun device. */ + configure_device(ipfd, tapif, ip); + + /* Expect Guest to handle everything except UFO */ + add_feature(dev, VIRTIO_NET_F_CSUM); + add_feature(dev, VIRTIO_NET_F_GUEST_CSUM); + add_feature(dev, VIRTIO_NET_F_GUEST_TSO4); + add_feature(dev, VIRTIO_NET_F_GUEST_TSO6); + add_feature(dev, VIRTIO_NET_F_GUEST_ECN); + add_feature(dev, VIRTIO_NET_F_HOST_TSO4); + add_feature(dev, VIRTIO_NET_F_HOST_TSO6); + add_feature(dev, VIRTIO_NET_F_HOST_ECN); + /* We handle indirect ring entries */ + add_feature(dev, VIRTIO_RING_F_INDIRECT_DESC); + set_config(dev, sizeof(conf), &conf); + + /* We don't need the socket any more; setup is done. */ + close(ipfd); + + devices.device_num++; + + if (bridging) + verbose("device %u: tun %s attached to bridge: %s\n", + devices.device_num, tapif, arg); + else + verbose("device %u: tun %s: %s\n", + devices.device_num, tapif, arg); +} +/*:*/ + +/* This hangs off device->priv. */ +struct vblk_info { + /* The size of the file. */ + off64_t len; + + /* The file descriptor for the file. */ + int fd; + +}; + +/*L:210 + * The Disk + * + * The disk only has one virtqueue, so it only has one thread. It is really + * simple: the Guest asks for a block number and we read or write that position + * in the file. + * + * Before we serviced each virtqueue in a separate thread, that was unacceptably + * slow: the Guest waits until the read is finished before running anything + * else, even if it could have been doing useful work. + * + * We could have used async I/O, except it's reputed to suck so hard that + * characters actually go missing from your code when you try to use it. + */ +static void blk_request(struct virtqueue *vq) +{ + struct vblk_info *vblk = vq->dev->priv; + unsigned int head, out_num, in_num, wlen; + int ret, i; + u8 *in; + struct virtio_blk_outhdr out; + struct iovec iov[vq->vring.num]; + off64_t off; + + /* + * Get the next request, where we normally wait. It triggers the + * interrupt to acknowledge previously serviced requests (if any). + */ + head = wait_for_vq_desc(vq, iov, &out_num, &in_num); + + /* Copy the output header from the front of the iov (adjusts iov) */ + iov_consume(iov, out_num, &out, sizeof(out)); + + /* Find and trim end of iov input array, for our status byte. */ + in = NULL; + for (i = out_num + in_num - 1; i >= out_num; i--) { + if (iov[i].iov_len > 0) { + in = iov[i].iov_base + iov[i].iov_len - 1; + iov[i].iov_len--; + break; + } + } + if (!in) + errx(1, "Bad virtblk cmd with no room for status"); + + /* + * For historical reasons, block operations are expressed in 512 byte + * "sectors". + */ + off = out.sector * 512; + + /* + * In general the virtio block driver is allowed to try SCSI commands. + * It'd be nice if we supported eject, for example, but we don't. + */ + if (out.type & VIRTIO_BLK_T_SCSI_CMD) { + fprintf(stderr, "Scsi commands unsupported\n"); + *in = VIRTIO_BLK_S_UNSUPP; + wlen = sizeof(*in); + } else if (out.type & VIRTIO_BLK_T_OUT) { + /* + * Write + * + * Move to the right location in the block file. This can fail + * if they try to write past end. + */ + if (lseek64(vblk->fd, off, SEEK_SET) != off) + err(1, "Bad seek to sector %llu", out.sector); + + ret = writev(vblk->fd, iov, out_num); + verbose("WRITE to sector %llu: %i\n", out.sector, ret); + + /* + * Grr... Now we know how long the descriptor they sent was, we + * make sure they didn't try to write over the end of the block + * file (possibly extending it). + */ + if (ret > 0 && off + ret > vblk->len) { + /* Trim it back to the correct length */ + ftruncate64(vblk->fd, vblk->len); + /* Die, bad Guest, die. */ + errx(1, "Write past end %llu+%u", off, ret); + } + + wlen = sizeof(*in); + *in = (ret >= 0 ? VIRTIO_BLK_S_OK : VIRTIO_BLK_S_IOERR); + } else if (out.type & VIRTIO_BLK_T_FLUSH) { + /* Flush */ + ret = fdatasync(vblk->fd); + verbose("FLUSH fdatasync: %i\n", ret); + wlen = sizeof(*in); + *in = (ret >= 0 ? VIRTIO_BLK_S_OK : VIRTIO_BLK_S_IOERR); + } else { + /* + * Read + * + * Move to the right location in the block file. This can fail + * if they try to read past end. + */ + if (lseek64(vblk->fd, off, SEEK_SET) != off) + err(1, "Bad seek to sector %llu", out.sector); + + ret = readv(vblk->fd, iov + out_num, in_num); + if (ret >= 0) { + wlen = sizeof(*in) + ret; + *in = VIRTIO_BLK_S_OK; + } else { + wlen = sizeof(*in); + *in = VIRTIO_BLK_S_IOERR; + } + } + + /* Finished that request. */ + add_used(vq, head, wlen); +} + +/*L:198 This actually sets up a virtual block device. */ +static void setup_block_file(const char *filename) +{ + struct device *dev; + struct vblk_info *vblk; + struct virtio_blk_config conf; + + /* Creat the device. */ + dev = new_device("block", VIRTIO_ID_BLOCK); + + /* The device has one virtqueue, where the Guest places requests. */ + add_virtqueue(dev, VIRTQUEUE_NUM, blk_request); + + /* Allocate the room for our own bookkeeping */ + vblk = dev->priv = malloc(sizeof(*vblk)); + + /* First we open the file and store the length. */ + vblk->fd = open_or_die(filename, O_RDWR|O_LARGEFILE); + vblk->len = lseek64(vblk->fd, 0, SEEK_END); + + /* We support FLUSH. */ + add_feature(dev, VIRTIO_BLK_F_FLUSH); + + /* Tell Guest how many sectors this device has. */ + conf.capacity = cpu_to_le64(vblk->len / 512); + + /* + * Tell Guest not to put in too many descriptors at once: two are used + * for the in and out elements. + */ + add_feature(dev, VIRTIO_BLK_F_SEG_MAX); + conf.seg_max = cpu_to_le32(VIRTQUEUE_NUM - 2); + + /* Don't try to put whole struct: we have 8 bit limit. */ + set_config(dev, offsetof(struct virtio_blk_config, geometry), &conf); + + verbose("device %u: virtblock %llu sectors\n", + ++devices.device_num, le64_to_cpu(conf.capacity)); +} + +/*L:211 + * Our random number generator device reads from /dev/random into the Guest's + * input buffers. The usual case is that the Guest doesn't want random numbers + * and so has no buffers although /dev/random is still readable, whereas + * console is the reverse. + * + * The same logic applies, however. + */ +struct rng_info { + int rfd; +}; + +static void rng_input(struct virtqueue *vq) +{ + int len; + unsigned int head, in_num, out_num, totlen = 0; + struct rng_info *rng_info = vq->dev->priv; + struct iovec iov[vq->vring.num]; + + /* First we need a buffer from the Guests's virtqueue. */ + head = wait_for_vq_desc(vq, iov, &out_num, &in_num); + if (out_num) + errx(1, "Output buffers in rng?"); + + /* + * Just like the console write, we loop to cover the whole iovec. + * In this case, short reads actually happen quite a bit. + */ + while (!iov_empty(iov, in_num)) { + len = readv(rng_info->rfd, iov, in_num); + if (len <= 0) + err(1, "Read from /dev/random gave %i", len); + iov_consume(iov, in_num, NULL, len); + totlen += len; + } + + /* Tell the Guest about the new input. */ + add_used(vq, head, totlen); +} + +/*L:199 + * This creates a "hardware" random number device for the Guest. + */ +static void setup_rng(void) +{ + struct device *dev; + struct rng_info *rng_info = malloc(sizeof(*rng_info)); + + /* Our device's privat info simply contains the /dev/random fd. */ + rng_info->rfd = open_or_die("/dev/random", O_RDONLY); + + /* Create the new device. */ + dev = new_device("rng", VIRTIO_ID_RNG); + dev->priv = rng_info; + + /* The device has one virtqueue, where the Guest places inbufs. */ + add_virtqueue(dev, VIRTQUEUE_NUM, rng_input); + + verbose("device %u: rng\n", devices.device_num++); +} +/* That's the end of device setup. */ + +/*L:230 Reboot is pretty easy: clean up and exec() the Launcher afresh. */ +static void __attribute__((noreturn)) restart_guest(void) +{ + unsigned int i; + + /* + * Since we don't track all open fds, we simply close everything beyond + * stderr. + */ + for (i = 3; i < FD_SETSIZE; i++) + close(i); + + /* Reset all the devices (kills all threads). */ + cleanup_devices(); + + execv(main_args[0], main_args); + err(1, "Could not exec %s", main_args[0]); +} + +/*L:220 + * Finally we reach the core of the Launcher which runs the Guest, serves + * its input and output, and finally, lays it to rest. + */ +static void __attribute__((noreturn)) run_guest(void) +{ + for (;;) { + unsigned long notify_addr; + int readval; + + /* We read from the /dev/lguest device to run the Guest. */ + readval = pread(lguest_fd, ¬ify_addr, + sizeof(notify_addr), cpu_id); + + /* One unsigned long means the Guest did HCALL_NOTIFY */ + if (readval == sizeof(notify_addr)) { + verbose("Notify on address %#lx\n", notify_addr); + handle_output(notify_addr); + /* ENOENT means the Guest died. Reading tells us why. */ + } else if (errno == ENOENT) { + char reason[1024] = { 0 }; + pread(lguest_fd, reason, sizeof(reason)-1, cpu_id); + errx(1, "%s", reason); + /* ERESTART means that we need to reboot the guest */ + } else if (errno == ERESTART) { + restart_guest(); + /* Anything else means a bug or incompatible change. */ + } else + err(1, "Running guest failed"); + } +} +/*L:240 + * This is the end of the Launcher. The good news: we are over halfway + * through! The bad news: the most fiendish part of the code still lies ahead + * of us. + * + * Are you ready? Take a deep breath and join me in the core of the Host, in + * "make Host". +:*/ + +static struct option opts[] = { + { "verbose", 0, NULL, 'v' }, + { "tunnet", 1, NULL, 't' }, + { "block", 1, NULL, 'b' }, + { "rng", 0, NULL, 'r' }, + { "initrd", 1, NULL, 'i' }, + { "username", 1, NULL, 'u' }, + { "chroot", 1, NULL, 'c' }, + { NULL }, +}; +static void usage(void) +{ + errx(1, "Usage: lguest [--verbose] " + "[--tunnet=(:|bridge::)\n" + "|--block=|--initrd=]...\n" + " vmlinux [args...]"); +} + +/*L:105 The main routine is where the real work begins: */ +int main(int argc, char *argv[]) +{ + /* Memory, code startpoint and size of the (optional) initrd. */ + unsigned long mem = 0, start, initrd_size = 0; + /* Two temporaries. */ + int i, c; + /* The boot information for the Guest. */ + struct boot_params *boot; + /* If they specify an initrd file to load. */ + const char *initrd_name = NULL; + + /* Password structure for initgroups/setres[gu]id */ + struct passwd *user_details = NULL; + + /* Directory to chroot to */ + char *chroot_path = NULL; + + /* Save the args: we "reboot" by execing ourselves again. */ + main_args = argv; + + /* + * First we initialize the device list. We keep a pointer to the last + * device, and the next interrupt number to use for devices (1: + * remember that 0 is used by the timer). + */ + devices.lastdev = NULL; + devices.next_irq = 1; + + /* We're CPU 0. In fact, that's the only CPU possible right now. */ + cpu_id = 0; + + /* + * We need to know how much memory so we can set up the device + * descriptor and memory pages for the devices as we parse the command + * line. So we quickly look through the arguments to find the amount + * of memory now. + */ + for (i = 1; i < argc; i++) { + if (argv[i][0] != '-') { + mem = atoi(argv[i]) * 1024 * 1024; + /* + * We start by mapping anonymous pages over all of + * guest-physical memory range. This fills it with 0, + * and ensures that the Guest won't be killed when it + * tries to access it. + */ + guest_base = map_zeroed_pages(mem / getpagesize() + + DEVICE_PAGES); + guest_limit = mem; + guest_max = mem + DEVICE_PAGES*getpagesize(); + devices.descpage = get_pages(1); + break; + } + } + + /* The options are fairly straight-forward */ + while ((c = getopt_long(argc, argv, "v", opts, NULL)) != EOF) { + switch (c) { + case 'v': + verbose = true; + break; + case 't': + setup_tun_net(optarg); + break; + case 'b': + setup_block_file(optarg); + break; + case 'r': + setup_rng(); + break; + case 'i': + initrd_name = optarg; + break; + case 'u': + user_details = getpwnam(optarg); + if (!user_details) + err(1, "getpwnam failed, incorrect username?"); + break; + case 'c': + chroot_path = optarg; + break; + default: + warnx("Unknown argument %s", argv[optind]); + usage(); + } + } + /* + * After the other arguments we expect memory and kernel image name, + * followed by command line arguments for the kernel. + */ + if (optind + 2 > argc) + usage(); + + verbose("Guest base is at %p\n", guest_base); + + /* We always have a console device */ + setup_console(); + + /* Now we load the kernel */ + start = load_kernel(open_or_die(argv[optind+1], O_RDONLY)); + + /* Boot information is stashed at physical address 0 */ + boot = from_guest_phys(0); + + /* Map the initrd image if requested (at top of physical memory) */ + if (initrd_name) { + initrd_size = load_initrd(initrd_name, mem); + /* + * These are the location in the Linux boot header where the + * start and size of the initrd are expected to be found. + */ + boot->hdr.ramdisk_image = mem - initrd_size; + boot->hdr.ramdisk_size = initrd_size; + /* The bootloader type 0xFF means "unknown"; that's OK. */ + boot->hdr.type_of_loader = 0xFF; + } + + /* + * The Linux boot header contains an "E820" memory map: ours is a + * simple, single region. + */ + boot->e820_entries = 1; + boot->e820_map[0] = ((struct e820entry) { 0, mem, E820_RAM }); + /* + * The boot header contains a command line pointer: we put the command + * line after the boot header. + */ + boot->hdr.cmd_line_ptr = to_guest_phys(boot + 1); + /* We use a simple helper to copy the arguments separated by spaces. */ + concat((char *)(boot + 1), argv+optind+2); + + /* Set kernel alignment to 16M (CONFIG_PHYSICAL_ALIGN) */ + boot->hdr.kernel_alignment = 0x1000000; + + /* Boot protocol version: 2.07 supports the fields for lguest. */ + boot->hdr.version = 0x207; + + /* The hardware_subarch value of "1" tells the Guest it's an lguest. */ + boot->hdr.hardware_subarch = 1; + + /* Tell the entry path not to try to reload segment registers. */ + boot->hdr.loadflags |= KEEP_SEGMENTS; + + /* We tell the kernel to initialize the Guest. */ + tell_kernel(start); + + /* Ensure that we terminate if a device-servicing child dies. */ + signal(SIGCHLD, kill_launcher); + + /* If we exit via err(), this kills all the threads, restores tty. */ + atexit(cleanup_devices); + + /* If requested, chroot to a directory */ + if (chroot_path) { + if (chroot(chroot_path) != 0) + err(1, "chroot(\"%s\") failed", chroot_path); + + if (chdir("/") != 0) + err(1, "chdir(\"/\") failed"); + + verbose("chroot done\n"); + } + + /* If requested, drop privileges */ + if (user_details) { + uid_t u; + gid_t g; + + u = user_details->pw_uid; + g = user_details->pw_gid; + + if (initgroups(user_details->pw_name, g) != 0) + err(1, "initgroups failed"); + + if (setresgid(g, g, g) != 0) + err(1, "setresgid failed"); + + if (setresuid(u, u, u) != 0) + err(1, "setresuid failed"); + + verbose("Dropping privileges completed\n"); + } + + /* Finally, run the Guest. This doesn't return. */ + run_guest(); +} +/*:*/ + +/*M:999 + * Mastery is done: you now know everything I do. + * + * But surely you have seen code, features and bugs in your wanderings which + * you now yearn to attack? That is the real game, and I look forward to you + * patching and forking lguest into the Your-Name-Here-visor. + * + * Farewell, and good coding! + * Rusty Russell. + */ diff --git a/tools/lguest/lguest.txt b/tools/lguest/lguest.txt new file mode 100644 index 000000000..06e1f4649 --- /dev/null +++ b/tools/lguest/lguest.txt @@ -0,0 +1,125 @@ + __ + (___()'`; Rusty's Remarkably Unreliable Guide to Lguest + /, /` - or, A Young Coder's Illustrated Hypervisor + \\"--\\ http://lguest.ozlabs.org + +Lguest is designed to be a minimal 32-bit x86 hypervisor for the Linux kernel, +for Linux developers and users to experiment with virtualization with the +minimum of complexity. Nonetheless, it should have sufficient features to +make it useful for specific tasks, and, of course, you are encouraged to fork +and enhance it (see drivers/lguest/README). + +Features: + +- Kernel module which runs in a normal kernel. +- Simple I/O model for communication. +- Simple program to create new guests. +- Logo contains cute puppies: http://lguest.ozlabs.org + +Developer features: + +- Fun to hack on. +- No ABI: being tied to a specific kernel anyway, you can change anything. +- Many opportunities for improvement or feature implementation. + +Running Lguest: + +- The easiest way to run lguest is to use same kernel as guest and host. + You can configure them differently, but usually it's easiest not to. + + You will need to configure your kernel with the following options: + + "Processor type and features": + "Paravirtualized guest support" = Y + "Lguest guest support" = Y + "High Memory Support" = off/4GB + "Alignment value to which kernel should be aligned" = 0x100000 + (CONFIG_PARAVIRT=y, CONFIG_LGUEST_GUEST=y, CONFIG_HIGHMEM64G=n and + CONFIG_PHYSICAL_ALIGN=0x100000) + + "Device Drivers": + "Block devices" + "Virtio block driver" = M/Y + "Network device support" + "Universal TUN/TAP device driver support" = M/Y + "Virtio network driver" = M/Y + (CONFIG_VIRTIO_BLK=m, CONFIG_VIRTIO_NET=m and CONFIG_TUN=m) + + "Virtualization" + "Linux hypervisor example code" = M/Y + (CONFIG_LGUEST=m) + +- A tool called "lguest" is available in this directory: type "make" + to build it. If you didn't build your kernel in-tree, use "make + O=". + +- Create or find a root disk image. There are several useful ones + around, such as the xm-test tiny root image at + http://xm-test.xensource.com/ramdisks/initrd-1.1-i386.img + + For more serious work, I usually use a distribution ISO image and + install it under qemu, then make multiple copies: + + dd if=/dev/zero of=rootfile bs=1M count=2048 + qemu -cdrom image.iso -hda rootfile -net user -net nic -boot d + + Make sure that you install a getty on /dev/hvc0 if you want to log in on the + console! + +- "modprobe lg" if you built it as a module. + +- Run an lguest as root: + + tools/lguest/lguest 64 vmlinux --tunnet=192.168.19.1 \ + --block=rootfile root=/dev/vda + + Explanation: + 64: the amount of memory to use, in MB. + + vmlinux: the kernel image found in the top of your build directory. You + can also use a standard bzImage. + + --tunnet=192.168.19.1: configures a "tap" device for networking with this + IP address. + + --block=rootfile: a file or block device which becomes /dev/vda + inside the guest. + + root=/dev/vda: this (and anything else on the command line) are + kernel boot parameters. + +- Configuring networking. I usually have the host masquerade, using + "iptables -t nat -A POSTROUTING -o eth0 -j MASQUERADE" and "echo 1 > + /proc/sys/net/ipv4/ip_forward". In this example, I would configure + eth0 inside the guest at 192.168.19.2. + + Another method is to bridge the tap device to an external interface + using --tunnet=bridge:, and perhaps run dhcp on the guest + to obtain an IP address. The bridge needs to be configured first: + this option simply adds the tap interface to it. + + A simple example on my system: + + ifconfig eth0 0.0.0.0 + brctl addbr lg0 + ifconfig lg0 up + brctl addif lg0 eth0 + dhclient lg0 + + Then use --tunnet=bridge:lg0 when launching the guest. + + See: + + http://www.linuxfoundation.org/collaborate/workgroups/networking/bridge + + for general information on how to get bridging to work. + +- Random number generation. Using the --rng option will provide a + /dev/hwrng in the guest that will read from the host's /dev/random. + Use this option in conjunction with rng-tools (see ../hw_random.txt) + to provide entropy to the guest kernel's /dev/random. + +There is a helpful mailing list at http://ozlabs.org/mailman/listinfo/lguest + +Good luck! +Rusty Russell rusty@rustcorp.com.au. diff --git a/tools/lib/lk/Makefile b/tools/lib/lk/Makefile new file mode 100644 index 000000000..926cbf3ef --- /dev/null +++ b/tools/lib/lk/Makefile @@ -0,0 +1,35 @@ +include ../../scripts/Makefile.include + +# guard against environment variables +LIB_H= +LIB_OBJS= + +LIB_H += debugfs.h + +LIB_OBJS += $(OUTPUT)debugfs.o + +LIBFILE = liblk.a + +CFLAGS = -ggdb3 -Wall -Wextra -std=gnu99 -Werror -O6 -D_FORTIFY_SOURCE=2 $(EXTRA_WARNINGS) $(EXTRA_CFLAGS) -fPIC +EXTLIBS = -lpthread -lrt -lelf -lm +ALL_CFLAGS = $(CFLAGS) $(BASIC_CFLAGS) -D_LARGEFILE64_SOURCE -D_FILE_OFFSET_BITS=64 +ALL_LDFLAGS = $(LDFLAGS) + +RM = rm -f + +$(LIBFILE): $(LIB_OBJS) + $(QUIET_AR)$(RM) $@ && $(AR) rcs $(OUTPUT)$@ $(LIB_OBJS) + +$(LIB_OBJS): $(LIB_H) + +$(OUTPUT)%.o: %.c + $(QUIET_CC)$(CC) -o $@ -c $(ALL_CFLAGS) $< +$(OUTPUT)%.s: %.c + $(QUIET_CC)$(CC) -S $(ALL_CFLAGS) $< +$(OUTPUT)%.o: %.S + $(QUIET_CC)$(CC) -o $@ -c $(ALL_CFLAGS) $< + +clean: + $(RM) $(LIB_OBJS) $(LIBFILE) + +.PHONY: clean diff --git a/tools/lib/lk/debugfs.c b/tools/lib/lk/debugfs.c new file mode 100644 index 000000000..7c4347962 --- /dev/null +++ b/tools/lib/lk/debugfs.c @@ -0,0 +1,100 @@ +#include +#include +#include +#include +#include +#include +#include +#include + +#include "debugfs.h" + +char debugfs_mountpoint[PATH_MAX + 1] = "/sys/kernel/debug"; + +static const char * const debugfs_known_mountpoints[] = { + "/sys/kernel/debug/", + "/debug/", + 0, +}; + +static bool debugfs_found; + +/* find the path to the mounted debugfs */ +const char *debugfs_find_mountpoint(void) +{ + const char * const *ptr; + char type[100]; + FILE *fp; + + if (debugfs_found) + return (const char *)debugfs_mountpoint; + + ptr = debugfs_known_mountpoints; + while (*ptr) { + if (debugfs_valid_mountpoint(*ptr) == 0) { + debugfs_found = true; + strcpy(debugfs_mountpoint, *ptr); + return debugfs_mountpoint; + } + ptr++; + } + + /* give up and parse /proc/mounts */ + fp = fopen("/proc/mounts", "r"); + if (fp == NULL) + return NULL; + + while (fscanf(fp, "%*s %" STR(PATH_MAX) "s %99s %*s %*d %*d\n", + debugfs_mountpoint, type) == 2) { + if (strcmp(type, "debugfs") == 0) + break; + } + fclose(fp); + + if (strcmp(type, "debugfs") != 0) + return NULL; + + debugfs_found = true; + + return debugfs_mountpoint; +} + +/* verify that a mountpoint is actually a debugfs instance */ + +int debugfs_valid_mountpoint(const char *debugfs) +{ + struct statfs st_fs; + + if (statfs(debugfs, &st_fs) < 0) + return -ENOENT; + else if (st_fs.f_type != (long) DEBUGFS_MAGIC) + return -ENOENT; + + return 0; +} + +/* mount the debugfs somewhere if it's not mounted */ +char *debugfs_mount(const char *mountpoint) +{ + /* see if it's already mounted */ + if (debugfs_find_mountpoint()) + goto out; + + /* if not mounted and no argument */ + if (mountpoint == NULL) { + /* see if environment variable set */ + mountpoint = getenv(PERF_DEBUGFS_ENVIRONMENT); + /* if no environment variable, use default */ + if (mountpoint == NULL) + mountpoint = "/sys/kernel/debug"; + } + + if (mount(NULL, mountpoint, "debugfs", 0, NULL) < 0) + return NULL; + + /* save the mountpoint */ + debugfs_found = true; + strncpy(debugfs_mountpoint, mountpoint, sizeof(debugfs_mountpoint)); +out: + return debugfs_mountpoint; +} diff --git a/tools/lib/lk/debugfs.h b/tools/lib/lk/debugfs.h new file mode 100644 index 000000000..935c59bdb --- /dev/null +++ b/tools/lib/lk/debugfs.h @@ -0,0 +1,29 @@ +#ifndef __LK_DEBUGFS_H__ +#define __LK_DEBUGFS_H__ + +#define _STR(x) #x +#define STR(x) _STR(x) + +/* + * On most systems would have given us this, but not on some systems + * (e.g. GNU/Hurd). + */ +#ifndef PATH_MAX +#define PATH_MAX 4096 +#endif + +#ifndef DEBUGFS_MAGIC +#define DEBUGFS_MAGIC 0x64626720 +#endif + +#ifndef PERF_DEBUGFS_ENVIRONMENT +#define PERF_DEBUGFS_ENVIRONMENT "PERF_DEBUGFS_DIR" +#endif + +const char *debugfs_find_mountpoint(void); +int debugfs_valid_mountpoint(const char *debugfs); +char *debugfs_mount(const char *mountpoint); + +extern char debugfs_mountpoint[]; + +#endif /* __LK_DEBUGFS_H__ */ diff --git a/tools/lib/traceevent/.gitignore b/tools/lib/traceevent/.gitignore new file mode 100644 index 000000000..35f56be5a --- /dev/null +++ b/tools/lib/traceevent/.gitignore @@ -0,0 +1 @@ +TRACEEVENT-CFLAGS diff --git a/tools/lib/traceevent/Makefile b/tools/lib/traceevent/Makefile new file mode 100644 index 000000000..0b0a90787 --- /dev/null +++ b/tools/lib/traceevent/Makefile @@ -0,0 +1,319 @@ +# trace-cmd version +EP_VERSION = 1 +EP_PATCHLEVEL = 1 +EP_EXTRAVERSION = 0 + +# file format version +FILE_VERSION = 6 + +MAKEFLAGS += --no-print-directory + + +# Makefiles suck: This macro sets a default value of $(2) for the +# variable named by $(1), unless the variable has been set by +# environment or command line. This is necessary for CC and AR +# because make sets default values, so the simpler ?= approach +# won't work as expected. +define allow-override + $(if $(or $(findstring environment,$(origin $(1))),\ + $(findstring command line,$(origin $(1)))),,\ + $(eval $(1) = $(2))) +endef + +# Allow setting CC and AR, or setting CROSS_COMPILE as a prefix. +$(call allow-override,CC,$(CROSS_COMPILE)gcc) +$(call allow-override,AR,$(CROSS_COMPILE)ar) + +EXT = -std=gnu99 +INSTALL = install + +# Use DESTDIR for installing into a different root directory. +# This is useful for building a package. The program will be +# installed in this directory as if it was the root directory. +# Then the build tool can move it later. +DESTDIR ?= +DESTDIR_SQ = '$(subst ','\'',$(DESTDIR))' + +prefix ?= /usr/local +bindir_relative = bin +bindir = $(prefix)/$(bindir_relative) +man_dir = $(prefix)/share/man +man_dir_SQ = '$(subst ','\'',$(man_dir))' +html_install = $(prefix)/share/kernelshark/html +html_install_SQ = '$(subst ','\'',$(html_install))' +img_install = $(prefix)/share/kernelshark/html/images +img_install_SQ = '$(subst ','\'',$(img_install))' + +export man_dir man_dir_SQ html_install html_install_SQ INSTALL +export img_install img_install_SQ +export DESTDIR DESTDIR_SQ + +# copy a bit from Linux kbuild + +ifeq ("$(origin V)", "command line") + VERBOSE = $(V) +endif +ifndef VERBOSE + VERBOSE = 0 +endif + +ifeq ("$(origin O)", "command line") + BUILD_OUTPUT := $(O) +endif + +ifeq ($(BUILD_SRC),) +ifneq ($(BUILD_OUTPUT),) + +define build_output + $(if $(VERBOSE:1=),@)$(MAKE) -C $(BUILD_OUTPUT) \ + BUILD_SRC=$(CURDIR) -f $(CURDIR)/Makefile $1 +endef + +saved-output := $(BUILD_OUTPUT) +BUILD_OUTPUT := $(shell cd $(BUILD_OUTPUT) && /bin/pwd) +$(if $(BUILD_OUTPUT),, \ + $(error output directory "$(saved-output)" does not exist)) + +all: sub-make + +gui: force + $(call build_output, all_cmd) + +$(filter-out gui,$(MAKECMDGOALS)): sub-make + +sub-make: force + $(call build_output, $(MAKECMDGOALS)) + + +# Leave processing to above invocation of make +skip-makefile := 1 + +endif # BUILD_OUTPUT +endif # BUILD_SRC + +# We process the rest of the Makefile if this is the final invocation of make +ifeq ($(skip-makefile),) + +srctree := $(if $(BUILD_SRC),$(BUILD_SRC),$(CURDIR)) +objtree := $(CURDIR) +src := $(srctree) +obj := $(objtree) + +export prefix bindir src obj + +# Shell quotes +bindir_SQ = $(subst ','\'',$(bindir)) +bindir_relative_SQ = $(subst ','\'',$(bindir_relative)) + +LIB_FILE = libtraceevent.a libtraceevent.so + +CONFIG_INCLUDES = +CONFIG_LIBS = +CONFIG_FLAGS = + +VERSION = $(EP_VERSION) +PATCHLEVEL = $(EP_PATCHLEVEL) +EXTRAVERSION = $(EP_EXTRAVERSION) + +OBJ = $@ +N = + +export Q VERBOSE + +EVENT_PARSE_VERSION = $(EP_VERSION).$(EP_PATCHLEVEL).$(EP_EXTRAVERSION) + +INCLUDES = -I. $(CONFIG_INCLUDES) + +# Set compile option CFLAGS if not set elsewhere +CFLAGS ?= -g -Wall + +# Append required CFLAGS +override CFLAGS += $(CONFIG_FLAGS) $(INCLUDES) $(PLUGIN_DIR_SQ) +override CFLAGS += $(udis86-flags) -D_GNU_SOURCE + +ifeq ($(VERBOSE),1) + Q = + print_compile = + print_app_build = + print_fpic_compile = + print_shared_lib_compile = + print_plugin_obj_compile = + print_plugin_build = + print_install = +else + Q = @ + print_compile = echo ' CC '$(OBJ); + print_app_build = echo ' BUILD '$(OBJ); + print_fpic_compile = echo ' CC FPIC '$(OBJ); + print_shared_lib_compile = echo ' BUILD SHARED LIB '$(OBJ); + print_plugin_obj_compile = echo ' CC PLUGIN OBJ '$(OBJ); + print_plugin_build = echo ' CC PLUGI '$(OBJ); + print_static_lib_build = echo ' BUILD STATIC LIB '$(OBJ); + print_install = echo ' INSTALL '$1' to $(DESTDIR_SQ)$2'; +endif + +do_fpic_compile = \ + ($(print_fpic_compile) \ + $(CC) -c $(CFLAGS) $(EXT) -fPIC $< -o $@) + +do_app_build = \ + ($(print_app_build) \ + $(CC) $^ -rdynamic -o $@ $(CONFIG_LIBS) $(LIBS)) + +do_compile_shared_library = \ + ($(print_shared_lib_compile) \ + $(CC) --shared $^ -o $@) + +do_compile_plugin_obj = \ + ($(print_plugin_obj_compile) \ + $(CC) -c $(CFLAGS) -fPIC -o $@ $<) + +do_plugin_build = \ + ($(print_plugin_build) \ + $(CC) $(CFLAGS) -shared -nostartfiles -o $@ $<) + +do_build_static_lib = \ + ($(print_static_lib_build) \ + $(RM) $@; $(AR) rcs $@ $^) + + +define do_compile + $(print_compile) \ + $(CC) -c $(CFLAGS) $(EXT) $< -o $(obj)/$@; +endef + +$(obj)/%.o: $(src)/%.c + $(Q)$(call do_compile) + +%.o: $(src)/%.c + $(Q)$(call do_compile) + +PEVENT_LIB_OBJS = event-parse.o trace-seq.o parse-filter.o parse-utils.o + +ALL_OBJS = $(PEVENT_LIB_OBJS) + +CMD_TARGETS = $(LIB_FILE) + +TARGETS = $(CMD_TARGETS) + + +all: all_cmd + +all_cmd: $(CMD_TARGETS) + +libtraceevent.so: $(PEVENT_LIB_OBJS) + $(Q)$(do_compile_shared_library) + +libtraceevent.a: $(PEVENT_LIB_OBJS) + $(Q)$(do_build_static_lib) + +$(PEVENT_LIB_OBJS): %.o: $(src)/%.c TRACEEVENT-CFLAGS + $(Q)$(do_fpic_compile) + +define make_version.h + (echo '/* This file is automatically generated. Do not modify. */'; \ + echo \#define VERSION_CODE $(shell \ + expr $(VERSION) \* 256 + $(PATCHLEVEL)); \ + echo '#define EXTRAVERSION ' $(EXTRAVERSION); \ + echo '#define VERSION_STRING "'$(VERSION).$(PATCHLEVEL).$(EXTRAVERSION)'"'; \ + echo '#define FILE_VERSION '$(FILE_VERSION); \ + ) > $1 +endef + +define update_version.h + ($(call make_version.h, $@.tmp); \ + if [ -r $@ ] && cmp -s $@ $@.tmp; then \ + rm -f $@.tmp; \ + else \ + echo ' UPDATE $@'; \ + mv -f $@.tmp $@; \ + fi); +endef + +ep_version.h: force + $(Q)$(N)$(call update_version.h) + +VERSION_FILES = ep_version.h + +define update_dir + (echo $1 > $@.tmp; \ + if [ -r $@ ] && cmp -s $@ $@.tmp; then \ + rm -f $@.tmp; \ + else \ + echo ' UPDATE $@'; \ + mv -f $@.tmp $@; \ + fi); +endef + +## make deps + +all_objs := $(sort $(ALL_OBJS)) +all_deps := $(all_objs:%.o=.%.d) + +# let .d file also depends on the source and header files +define check_deps + @set -e; $(RM) $@; \ + $(CC) -MM $(CFLAGS) $< > $@.$$$$; \ + sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \ + $(RM) $@.$$$$ +endef + +$(gui_deps): ks_version.h +$(non_gui_deps): tc_version.h + +$(all_deps): .%.d: $(src)/%.c + $(Q)$(call check_deps) + +$(all_objs) : %.o : .%.d + +dep_includes := $(wildcard $(all_deps)) + +ifneq ($(dep_includes),) + include $(dep_includes) +endif + +### Detect environment changes +TRACK_CFLAGS = $(subst ','\'',$(CFLAGS)):$(ARCH):$(CROSS_COMPILE) + +TRACEEVENT-CFLAGS: force + @FLAGS='$(TRACK_CFLAGS)'; \ + if test x"$$FLAGS" != x"`cat TRACEEVENT-CFLAGS 2>/dev/null`" ; then \ + echo 1>&2 " * new build flags or cross compiler"; \ + echo "$$FLAGS" >TRACEEVENT-CFLAGS; \ + fi + +tags: force + $(RM) tags + find . -name '*.[ch]' | xargs ctags --extra=+f --c-kinds=+px \ + --regex-c++='/_PE\(([^,)]*).*/PEVENT_ERRNO__\1/' + +TAGS: force + $(RM) TAGS + find . -name '*.[ch]' | xargs etags \ + --regex='/_PE(\([^,)]*\).*/PEVENT_ERRNO__\1/' + +define do_install + $(print_install) \ + if [ ! -d '$(DESTDIR_SQ)$2' ]; then \ + $(INSTALL) -d -m 755 '$(DESTDIR_SQ)$2'; \ + fi; \ + $(INSTALL) $1 '$(DESTDIR_SQ)$2' +endef + +install_lib: all_cmd install_plugins install_python + $(Q)$(call do_install,$(LIB_FILE),$(bindir_SQ)) + +install: install_lib + +clean: + $(RM) *.o *~ $(TARGETS) *.a *.so $(VERSION_FILES) .*.d + $(RM) TRACEEVENT-CFLAGS tags TAGS + +endif # skip-makefile + +PHONY += force +force: + +# Declare the contents of the .PHONY variable as phony. We keep that +# information in a variable so we can use it in if_changed and friends. +.PHONY: $(PHONY) diff --git a/tools/lib/traceevent/event-parse.c b/tools/lib/traceevent/event-parse.c new file mode 100644 index 000000000..82b0606dc --- /dev/null +++ b/tools/lib/traceevent/event-parse.c @@ -0,0 +1,5654 @@ +/* + * Copyright (C) 2009, 2010 Red Hat Inc, Steven Rostedt + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License (not later!) + * + * 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this program; if not, see + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + * + * The parts for function graph printing was taken and modified from the + * Linux Kernel that were written by + * - Copyright (C) 2009 Frederic Weisbecker, + * Frederic Weisbecker gave his permission to relicense the code to + * the Lesser General Public License. + */ +#include +#include +#include +#include +#include +#include +#include +#include + +#include "event-parse.h" +#include "event-utils.h" + +static const char *input_buf; +static unsigned long long input_buf_ptr; +static unsigned long long input_buf_siz; + +static int is_flag_field; +static int is_symbolic_field; + +static int show_warning = 1; + +#define do_warning(fmt, ...) \ + do { \ + if (show_warning) \ + warning(fmt, ##__VA_ARGS__); \ + } while (0) + +static void init_input_buf(const char *buf, unsigned long long size) +{ + input_buf = buf; + input_buf_siz = size; + input_buf_ptr = 0; +} + +const char *pevent_get_input_buf(void) +{ + return input_buf; +} + +unsigned long long pevent_get_input_buf_ptr(void) +{ + return input_buf_ptr; +} + +struct event_handler { + struct event_handler *next; + int id; + const char *sys_name; + const char *event_name; + pevent_event_handler_func func; + void *context; +}; + +struct pevent_func_params { + struct pevent_func_params *next; + enum pevent_func_arg_type type; +}; + +struct pevent_function_handler { + struct pevent_function_handler *next; + enum pevent_func_arg_type ret_type; + char *name; + pevent_func_handler func; + struct pevent_func_params *params; + int nr_args; +}; + +static unsigned long long +process_defined_func(struct trace_seq *s, void *data, int size, + struct event_format *event, struct print_arg *arg); + +static void free_func_handle(struct pevent_function_handler *func); + +/** + * pevent_buffer_init - init buffer for parsing + * @buf: buffer to parse + * @size: the size of the buffer + * + * For use with pevent_read_token(), this initializes the internal + * buffer that pevent_read_token() will parse. + */ +void pevent_buffer_init(const char *buf, unsigned long long size) +{ + init_input_buf(buf, size); +} + +void breakpoint(void) +{ + static int x; + x++; +} + +struct print_arg *alloc_arg(void) +{ + return calloc(1, sizeof(struct print_arg)); +} + +struct cmdline { + char *comm; + int pid; +}; + +static int cmdline_cmp(const void *a, const void *b) +{ + const struct cmdline *ca = a; + const struct cmdline *cb = b; + + if (ca->pid < cb->pid) + return -1; + if (ca->pid > cb->pid) + return 1; + + return 0; +} + +struct cmdline_list { + struct cmdline_list *next; + char *comm; + int pid; +}; + +static int cmdline_init(struct pevent *pevent) +{ + struct cmdline_list *cmdlist = pevent->cmdlist; + struct cmdline_list *item; + struct cmdline *cmdlines; + int i; + + cmdlines = malloc(sizeof(*cmdlines) * pevent->cmdline_count); + if (!cmdlines) + return -1; + + i = 0; + while (cmdlist) { + cmdlines[i].pid = cmdlist->pid; + cmdlines[i].comm = cmdlist->comm; + i++; + item = cmdlist; + cmdlist = cmdlist->next; + free(item); + } + + qsort(cmdlines, pevent->cmdline_count, sizeof(*cmdlines), cmdline_cmp); + + pevent->cmdlines = cmdlines; + pevent->cmdlist = NULL; + + return 0; +} + +static const char *find_cmdline(struct pevent *pevent, int pid) +{ + const struct cmdline *comm; + struct cmdline key; + + if (!pid) + return ""; + + if (!pevent->cmdlines && cmdline_init(pevent)) + return ""; + + key.pid = pid; + + comm = bsearch(&key, pevent->cmdlines, pevent->cmdline_count, + sizeof(*pevent->cmdlines), cmdline_cmp); + + if (comm) + return comm->comm; + return "<...>"; +} + +/** + * pevent_pid_is_registered - return if a pid has a cmdline registered + * @pevent: handle for the pevent + * @pid: The pid to check if it has a cmdline registered with. + * + * Returns 1 if the pid has a cmdline mapped to it + * 0 otherwise. + */ +int pevent_pid_is_registered(struct pevent *pevent, int pid) +{ + const struct cmdline *comm; + struct cmdline key; + + if (!pid) + return 1; + + if (!pevent->cmdlines && cmdline_init(pevent)) + return 0; + + key.pid = pid; + + comm = bsearch(&key, pevent->cmdlines, pevent->cmdline_count, + sizeof(*pevent->cmdlines), cmdline_cmp); + + if (comm) + return 1; + return 0; +} + +/* + * If the command lines have been converted to an array, then + * we must add this pid. This is much slower than when cmdlines + * are added before the array is initialized. + */ +static int add_new_comm(struct pevent *pevent, const char *comm, int pid) +{ + struct cmdline *cmdlines = pevent->cmdlines; + const struct cmdline *cmdline; + struct cmdline key; + + if (!pid) + return 0; + + /* avoid duplicates */ + key.pid = pid; + + cmdline = bsearch(&key, pevent->cmdlines, pevent->cmdline_count, + sizeof(*pevent->cmdlines), cmdline_cmp); + if (cmdline) { + errno = EEXIST; + return -1; + } + + cmdlines = realloc(cmdlines, sizeof(*cmdlines) * (pevent->cmdline_count + 1)); + if (!cmdlines) { + errno = ENOMEM; + return -1; + } + + cmdlines[pevent->cmdline_count].comm = strdup(comm); + if (!cmdlines[pevent->cmdline_count].comm) { + free(cmdlines); + errno = ENOMEM; + return -1; + } + + cmdlines[pevent->cmdline_count].pid = pid; + + if (cmdlines[pevent->cmdline_count].comm) + pevent->cmdline_count++; + + qsort(cmdlines, pevent->cmdline_count, sizeof(*cmdlines), cmdline_cmp); + pevent->cmdlines = cmdlines; + + return 0; +} + +/** + * pevent_register_comm - register a pid / comm mapping + * @pevent: handle for the pevent + * @comm: the command line to register + * @pid: the pid to map the command line to + * + * This adds a mapping to search for command line names with + * a given pid. The comm is duplicated. + */ +int pevent_register_comm(struct pevent *pevent, const char *comm, int pid) +{ + struct cmdline_list *item; + + if (pevent->cmdlines) + return add_new_comm(pevent, comm, pid); + + item = malloc(sizeof(*item)); + if (!item) + return -1; + + item->comm = strdup(comm); + if (!item->comm) { + free(item); + return -1; + } + item->pid = pid; + item->next = pevent->cmdlist; + + pevent->cmdlist = item; + pevent->cmdline_count++; + + return 0; +} + +struct func_map { + unsigned long long addr; + char *func; + char *mod; +}; + +struct func_list { + struct func_list *next; + unsigned long long addr; + char *func; + char *mod; +}; + +static int func_cmp(const void *a, const void *b) +{ + const struct func_map *fa = a; + const struct func_map *fb = b; + + if (fa->addr < fb->addr) + return -1; + if (fa->addr > fb->addr) + return 1; + + return 0; +} + +/* + * We are searching for a record in between, not an exact + * match. + */ +static int func_bcmp(const void *a, const void *b) +{ + const struct func_map *fa = a; + const struct func_map *fb = b; + + if ((fa->addr == fb->addr) || + + (fa->addr > fb->addr && + fa->addr < (fb+1)->addr)) + return 0; + + if (fa->addr < fb->addr) + return -1; + + return 1; +} + +static int func_map_init(struct pevent *pevent) +{ + struct func_list *funclist; + struct func_list *item; + struct func_map *func_map; + int i; + + func_map = malloc(sizeof(*func_map) * (pevent->func_count + 1)); + if (!func_map) + return -1; + + funclist = pevent->funclist; + + i = 0; + while (funclist) { + func_map[i].func = funclist->func; + func_map[i].addr = funclist->addr; + func_map[i].mod = funclist->mod; + i++; + item = funclist; + funclist = funclist->next; + free(item); + } + + qsort(func_map, pevent->func_count, sizeof(*func_map), func_cmp); + + /* + * Add a special record at the end. + */ + func_map[pevent->func_count].func = NULL; + func_map[pevent->func_count].addr = 0; + func_map[pevent->func_count].mod = NULL; + + pevent->func_map = func_map; + pevent->funclist = NULL; + + return 0; +} + +static struct func_map * +find_func(struct pevent *pevent, unsigned long long addr) +{ + struct func_map *func; + struct func_map key; + + if (!pevent->func_map) + func_map_init(pevent); + + key.addr = addr; + + func = bsearch(&key, pevent->func_map, pevent->func_count, + sizeof(*pevent->func_map), func_bcmp); + + return func; +} + +/** + * pevent_find_function - find a function by a given address + * @pevent: handle for the pevent + * @addr: the address to find the function with + * + * Returns a pointer to the function stored that has the given + * address. Note, the address does not have to be exact, it + * will select the function that would contain the address. + */ +const char *pevent_find_function(struct pevent *pevent, unsigned long long addr) +{ + struct func_map *map; + + map = find_func(pevent, addr); + if (!map) + return NULL; + + return map->func; +} + +/** + * pevent_find_function_address - find a function address by a given address + * @pevent: handle for the pevent + * @addr: the address to find the function with + * + * Returns the address the function starts at. This can be used in + * conjunction with pevent_find_function to print both the function + * name and the function offset. + */ +unsigned long long +pevent_find_function_address(struct pevent *pevent, unsigned long long addr) +{ + struct func_map *map; + + map = find_func(pevent, addr); + if (!map) + return 0; + + return map->addr; +} + +/** + * pevent_register_function - register a function with a given address + * @pevent: handle for the pevent + * @function: the function name to register + * @addr: the address the function starts at + * @mod: the kernel module the function may be in (NULL for none) + * + * This registers a function name with an address and module. + * The @func passed in is duplicated. + */ +int pevent_register_function(struct pevent *pevent, char *func, + unsigned long long addr, char *mod) +{ + struct func_list *item = malloc(sizeof(*item)); + + if (!item) + return -1; + + item->next = pevent->funclist; + item->func = strdup(func); + if (!item->func) + goto out_free; + + if (mod) { + item->mod = strdup(mod); + if (!item->mod) + goto out_free_func; + } else + item->mod = NULL; + item->addr = addr; + + pevent->funclist = item; + pevent->func_count++; + + return 0; + +out_free_func: + free(item->func); + item->func = NULL; +out_free: + free(item); + errno = ENOMEM; + return -1; +} + +/** + * pevent_print_funcs - print out the stored functions + * @pevent: handle for the pevent + * + * This prints out the stored functions. + */ +void pevent_print_funcs(struct pevent *pevent) +{ + int i; + + if (!pevent->func_map) + func_map_init(pevent); + + for (i = 0; i < (int)pevent->func_count; i++) { + printf("%016llx %s", + pevent->func_map[i].addr, + pevent->func_map[i].func); + if (pevent->func_map[i].mod) + printf(" [%s]\n", pevent->func_map[i].mod); + else + printf("\n"); + } +} + +struct printk_map { + unsigned long long addr; + char *printk; +}; + +struct printk_list { + struct printk_list *next; + unsigned long long addr; + char *printk; +}; + +static int printk_cmp(const void *a, const void *b) +{ + const struct printk_map *pa = a; + const struct printk_map *pb = b; + + if (pa->addr < pb->addr) + return -1; + if (pa->addr > pb->addr) + return 1; + + return 0; +} + +static int printk_map_init(struct pevent *pevent) +{ + struct printk_list *printklist; + struct printk_list *item; + struct printk_map *printk_map; + int i; + + printk_map = malloc(sizeof(*printk_map) * (pevent->printk_count + 1)); + if (!printk_map) + return -1; + + printklist = pevent->printklist; + + i = 0; + while (printklist) { + printk_map[i].printk = printklist->printk; + printk_map[i].addr = printklist->addr; + i++; + item = printklist; + printklist = printklist->next; + free(item); + } + + qsort(printk_map, pevent->printk_count, sizeof(*printk_map), printk_cmp); + + pevent->printk_map = printk_map; + pevent->printklist = NULL; + + return 0; +} + +static struct printk_map * +find_printk(struct pevent *pevent, unsigned long long addr) +{ + struct printk_map *printk; + struct printk_map key; + + if (!pevent->printk_map && printk_map_init(pevent)) + return NULL; + + key.addr = addr; + + printk = bsearch(&key, pevent->printk_map, pevent->printk_count, + sizeof(*pevent->printk_map), printk_cmp); + + return printk; +} + +/** + * pevent_register_print_string - register a string by its address + * @pevent: handle for the pevent + * @fmt: the string format to register + * @addr: the address the string was located at + * + * This registers a string by the address it was stored in the kernel. + * The @fmt passed in is duplicated. + */ +int pevent_register_print_string(struct pevent *pevent, char *fmt, + unsigned long long addr) +{ + struct printk_list *item = malloc(sizeof(*item)); + + if (!item) + return -1; + + item->next = pevent->printklist; + item->addr = addr; + + item->printk = strdup(fmt); + if (!item->printk) + goto out_free; + + pevent->printklist = item; + pevent->printk_count++; + + return 0; + +out_free: + free(item); + errno = ENOMEM; + return -1; +} + +/** + * pevent_print_printk - print out the stored strings + * @pevent: handle for the pevent + * + * This prints the string formats that were stored. + */ +void pevent_print_printk(struct pevent *pevent) +{ + int i; + + if (!pevent->printk_map) + printk_map_init(pevent); + + for (i = 0; i < (int)pevent->printk_count; i++) { + printf("%016llx %s\n", + pevent->printk_map[i].addr, + pevent->printk_map[i].printk); + } +} + +static struct event_format *alloc_event(void) +{ + return calloc(1, sizeof(struct event_format)); +} + +static int add_event(struct pevent *pevent, struct event_format *event) +{ + int i; + struct event_format **events = realloc(pevent->events, sizeof(event) * + (pevent->nr_events + 1)); + if (!events) + return -1; + + pevent->events = events; + + for (i = 0; i < pevent->nr_events; i++) { + if (pevent->events[i]->id > event->id) + break; + } + if (i < pevent->nr_events) + memmove(&pevent->events[i + 1], + &pevent->events[i], + sizeof(event) * (pevent->nr_events - i)); + + pevent->events[i] = event; + pevent->nr_events++; + + event->pevent = pevent; + + return 0; +} + +static int event_item_type(enum event_type type) +{ + switch (type) { + case EVENT_ITEM ... EVENT_SQUOTE: + return 1; + case EVENT_ERROR ... EVENT_DELIM: + default: + return 0; + } +} + +static void free_flag_sym(struct print_flag_sym *fsym) +{ + struct print_flag_sym *next; + + while (fsym) { + next = fsym->next; + free(fsym->value); + free(fsym->str); + free(fsym); + fsym = next; + } +} + +static void free_arg(struct print_arg *arg) +{ + struct print_arg *farg; + + if (!arg) + return; + + switch (arg->type) { + case PRINT_ATOM: + free(arg->atom.atom); + break; + case PRINT_FIELD: + free(arg->field.name); + break; + case PRINT_FLAGS: + free_arg(arg->flags.field); + free(arg->flags.delim); + free_flag_sym(arg->flags.flags); + break; + case PRINT_SYMBOL: + free_arg(arg->symbol.field); + free_flag_sym(arg->symbol.symbols); + break; + case PRINT_HEX: + free_arg(arg->hex.field); + free_arg(arg->hex.size); + break; + case PRINT_TYPE: + free(arg->typecast.type); + free_arg(arg->typecast.item); + break; + case PRINT_STRING: + case PRINT_BSTRING: + free(arg->string.string); + break; + case PRINT_DYNAMIC_ARRAY: + free(arg->dynarray.index); + break; + case PRINT_OP: + free(arg->op.op); + free_arg(arg->op.left); + free_arg(arg->op.right); + break; + case PRINT_FUNC: + while (arg->func.args) { + farg = arg->func.args; + arg->func.args = farg->next; + free_arg(farg); + } + break; + + case PRINT_NULL: + default: + break; + } + + free(arg); +} + +static enum event_type get_type(int ch) +{ + if (ch == '\n') + return EVENT_NEWLINE; + if (isspace(ch)) + return EVENT_SPACE; + if (isalnum(ch) || ch == '_') + return EVENT_ITEM; + if (ch == '\'') + return EVENT_SQUOTE; + if (ch == '"') + return EVENT_DQUOTE; + if (!isprint(ch)) + return EVENT_NONE; + if (ch == '(' || ch == ')' || ch == ',') + return EVENT_DELIM; + + return EVENT_OP; +} + +static int __read_char(void) +{ + if (input_buf_ptr >= input_buf_siz) + return -1; + + return input_buf[input_buf_ptr++]; +} + +static int __peek_char(void) +{ + if (input_buf_ptr >= input_buf_siz) + return -1; + + return input_buf[input_buf_ptr]; +} + +/** + * pevent_peek_char - peek at the next character that will be read + * + * Returns the next character read, or -1 if end of buffer. + */ +int pevent_peek_char(void) +{ + return __peek_char(); +} + +static int extend_token(char **tok, char *buf, int size) +{ + char *newtok = realloc(*tok, size); + + if (!newtok) { + free(*tok); + *tok = NULL; + return -1; + } + + if (!*tok) + strcpy(newtok, buf); + else + strcat(newtok, buf); + *tok = newtok; + + return 0; +} + +static enum event_type force_token(const char *str, char **tok); + +static enum event_type __read_token(char **tok) +{ + char buf[BUFSIZ]; + int ch, last_ch, quote_ch, next_ch; + int i = 0; + int tok_size = 0; + enum event_type type; + + *tok = NULL; + + + ch = __read_char(); + if (ch < 0) + return EVENT_NONE; + + type = get_type(ch); + if (type == EVENT_NONE) + return type; + + buf[i++] = ch; + + switch (type) { + case EVENT_NEWLINE: + case EVENT_DELIM: + if (asprintf(tok, "%c", ch) < 0) + return EVENT_ERROR; + + return type; + + case EVENT_OP: + switch (ch) { + case '-': + next_ch = __peek_char(); + if (next_ch == '>') { + buf[i++] = __read_char(); + break; + } + /* fall through */ + case '+': + case '|': + case '&': + case '>': + case '<': + last_ch = ch; + ch = __peek_char(); + if (ch != last_ch) + goto test_equal; + buf[i++] = __read_char(); + switch (last_ch) { + case '>': + case '<': + goto test_equal; + default: + break; + } + break; + case '!': + case '=': + goto test_equal; + default: /* what should we do instead? */ + break; + } + buf[i] = 0; + *tok = strdup(buf); + return type; + + test_equal: + ch = __peek_char(); + if (ch == '=') + buf[i++] = __read_char(); + goto out; + + case EVENT_DQUOTE: + case EVENT_SQUOTE: + /* don't keep quotes */ + i--; + quote_ch = ch; + last_ch = 0; + concat: + do { + if (i == (BUFSIZ - 1)) { + buf[i] = 0; + tok_size += BUFSIZ; + + if (extend_token(tok, buf, tok_size) < 0) + return EVENT_NONE; + i = 0; + } + last_ch = ch; + ch = __read_char(); + buf[i++] = ch; + /* the '\' '\' will cancel itself */ + if (ch == '\\' && last_ch == '\\') + last_ch = 0; + } while (ch != quote_ch || last_ch == '\\'); + /* remove the last quote */ + i--; + + /* + * For strings (double quotes) check the next token. + * If it is another string, concatinate the two. + */ + if (type == EVENT_DQUOTE) { + unsigned long long save_input_buf_ptr = input_buf_ptr; + + do { + ch = __read_char(); + } while (isspace(ch)); + if (ch == '"') + goto concat; + input_buf_ptr = save_input_buf_ptr; + } + + goto out; + + case EVENT_ERROR ... EVENT_SPACE: + case EVENT_ITEM: + default: + break; + } + + while (get_type(__peek_char()) == type) { + if (i == (BUFSIZ - 1)) { + buf[i] = 0; + tok_size += BUFSIZ; + + if (extend_token(tok, buf, tok_size) < 0) + return EVENT_NONE; + i = 0; + } + ch = __read_char(); + buf[i++] = ch; + } + + out: + buf[i] = 0; + if (extend_token(tok, buf, tok_size + i + 1) < 0) + return EVENT_NONE; + + if (type == EVENT_ITEM) { + /* + * Older versions of the kernel has a bug that + * creates invalid symbols and will break the mac80211 + * parsing. This is a work around to that bug. + * + * See Linux kernel commit: + * 811cb50baf63461ce0bdb234927046131fc7fa8b + */ + if (strcmp(*tok, "LOCAL_PR_FMT") == 0) { + free(*tok); + *tok = NULL; + return force_token("\"\%s\" ", tok); + } else if (strcmp(*tok, "STA_PR_FMT") == 0) { + free(*tok); + *tok = NULL; + return force_token("\" sta:%pM\" ", tok); + } else if (strcmp(*tok, "VIF_PR_FMT") == 0) { + free(*tok); + *tok = NULL; + return force_token("\" vif:%p(%d)\" ", tok); + } + } + + return type; +} + +static enum event_type force_token(const char *str, char **tok) +{ + const char *save_input_buf; + unsigned long long save_input_buf_ptr; + unsigned long long save_input_buf_siz; + enum event_type type; + + /* save off the current input pointers */ + save_input_buf = input_buf; + save_input_buf_ptr = input_buf_ptr; + save_input_buf_siz = input_buf_siz; + + init_input_buf(str, strlen(str)); + + type = __read_token(tok); + + /* reset back to original token */ + input_buf = save_input_buf; + input_buf_ptr = save_input_buf_ptr; + input_buf_siz = save_input_buf_siz; + + return type; +} + +static void free_token(char *tok) +{ + if (tok) + free(tok); +} + +static enum event_type read_token(char **tok) +{ + enum event_type type; + + for (;;) { + type = __read_token(tok); + if (type != EVENT_SPACE) + return type; + + free_token(*tok); + } + + /* not reached */ + *tok = NULL; + return EVENT_NONE; +} + +/** + * pevent_read_token - access to utilites to use the pevent parser + * @tok: The token to return + * + * This will parse tokens from the string given by + * pevent_init_data(). + * + * Returns the token type. + */ +enum event_type pevent_read_token(char **tok) +{ + return read_token(tok); +} + +/** + * pevent_free_token - free a token returned by pevent_read_token + * @token: the token to free + */ +void pevent_free_token(char *token) +{ + free_token(token); +} + +/* no newline */ +static enum event_type read_token_item(char **tok) +{ + enum event_type type; + + for (;;) { + type = __read_token(tok); + if (type != EVENT_SPACE && type != EVENT_NEWLINE) + return type; + free_token(*tok); + *tok = NULL; + } + + /* not reached */ + *tok = NULL; + return EVENT_NONE; +} + +static int test_type(enum event_type type, enum event_type expect) +{ + if (type != expect) { + do_warning("Error: expected type %d but read %d", + expect, type); + return -1; + } + return 0; +} + +static int test_type_token(enum event_type type, const char *token, + enum event_type expect, const char *expect_tok) +{ + if (type != expect) { + do_warning("Error: expected type %d but read %d", + expect, type); + return -1; + } + + if (strcmp(token, expect_tok) != 0) { + do_warning("Error: expected '%s' but read '%s'", + expect_tok, token); + return -1; + } + return 0; +} + +static int __read_expect_type(enum event_type expect, char **tok, int newline_ok) +{ + enum event_type type; + + if (newline_ok) + type = read_token(tok); + else + type = read_token_item(tok); + return test_type(type, expect); +} + +static int read_expect_type(enum event_type expect, char **tok) +{ + return __read_expect_type(expect, tok, 1); +} + +static int __read_expected(enum event_type expect, const char *str, + int newline_ok) +{ + enum event_type type; + char *token; + int ret; + + if (newline_ok) + type = read_token(&token); + else + type = read_token_item(&token); + + ret = test_type_token(type, token, expect, str); + + free_token(token); + + return ret; +} + +static int read_expected(enum event_type expect, const char *str) +{ + return __read_expected(expect, str, 1); +} + +static int read_expected_item(enum event_type expect, const char *str) +{ + return __read_expected(expect, str, 0); +} + +static char *event_read_name(void) +{ + char *token; + + if (read_expected(EVENT_ITEM, "name") < 0) + return NULL; + + if (read_expected(EVENT_OP, ":") < 0) + return NULL; + + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto fail; + + return token; + + fail: + free_token(token); + return NULL; +} + +static int event_read_id(void) +{ + char *token; + int id; + + if (read_expected_item(EVENT_ITEM, "ID") < 0) + return -1; + + if (read_expected(EVENT_OP, ":") < 0) + return -1; + + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto fail; + + id = strtoul(token, NULL, 0); + free_token(token); + return id; + + fail: + free_token(token); + return -1; +} + +static int field_is_string(struct format_field *field) +{ + if ((field->flags & FIELD_IS_ARRAY) && + (strstr(field->type, "char") || strstr(field->type, "u8") || + strstr(field->type, "s8"))) + return 1; + + return 0; +} + +static int field_is_dynamic(struct format_field *field) +{ + if (strncmp(field->type, "__data_loc", 10) == 0) + return 1; + + return 0; +} + +static int field_is_long(struct format_field *field) +{ + /* includes long long */ + if (strstr(field->type, "long")) + return 1; + + return 0; +} + +static unsigned int type_size(const char *name) +{ + /* This covers all FIELD_IS_STRING types. */ + static struct { + const char *type; + unsigned int size; + } table[] = { + { "u8", 1 }, + { "u16", 2 }, + { "u32", 4 }, + { "u64", 8 }, + { "s8", 1 }, + { "s16", 2 }, + { "s32", 4 }, + { "s64", 8 }, + { "char", 1 }, + { }, + }; + int i; + + for (i = 0; table[i].type; i++) { + if (!strcmp(table[i].type, name)) + return table[i].size; + } + + return 0; +} + +static int event_read_fields(struct event_format *event, struct format_field **fields) +{ + struct format_field *field = NULL; + enum event_type type; + char *token; + char *last_token; + int count = 0; + + do { + unsigned int size_dynamic = 0; + + type = read_token(&token); + if (type == EVENT_NEWLINE) { + free_token(token); + return count; + } + + count++; + + if (test_type_token(type, token, EVENT_ITEM, "field")) + goto fail; + free_token(token); + + type = read_token(&token); + /* + * The ftrace fields may still use the "special" name. + * Just ignore it. + */ + if (event->flags & EVENT_FL_ISFTRACE && + type == EVENT_ITEM && strcmp(token, "special") == 0) { + free_token(token); + type = read_token(&token); + } + + if (test_type_token(type, token, EVENT_OP, ":") < 0) + goto fail; + + free_token(token); + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto fail; + + last_token = token; + + field = calloc(1, sizeof(*field)); + if (!field) + goto fail; + + field->event = event; + + /* read the rest of the type */ + for (;;) { + type = read_token(&token); + if (type == EVENT_ITEM || + (type == EVENT_OP && strcmp(token, "*") == 0) || + /* + * Some of the ftrace fields are broken and have + * an illegal "." in them. + */ + (event->flags & EVENT_FL_ISFTRACE && + type == EVENT_OP && strcmp(token, ".") == 0)) { + + if (strcmp(token, "*") == 0) + field->flags |= FIELD_IS_POINTER; + + if (field->type) { + char *new_type; + new_type = realloc(field->type, + strlen(field->type) + + strlen(last_token) + 2); + if (!new_type) { + free(last_token); + goto fail; + } + field->type = new_type; + strcat(field->type, " "); + strcat(field->type, last_token); + free(last_token); + } else + field->type = last_token; + last_token = token; + continue; + } + + break; + } + + if (!field->type) { + do_warning("%s: no type found", __func__); + goto fail; + } + field->name = last_token; + + if (test_type(type, EVENT_OP)) + goto fail; + + if (strcmp(token, "[") == 0) { + enum event_type last_type = type; + char *brackets = token; + char *new_brackets; + int len; + + field->flags |= FIELD_IS_ARRAY; + + type = read_token(&token); + + if (type == EVENT_ITEM) + field->arraylen = strtoul(token, NULL, 0); + else + field->arraylen = 0; + + while (strcmp(token, "]") != 0) { + if (last_type == EVENT_ITEM && + type == EVENT_ITEM) + len = 2; + else + len = 1; + last_type = type; + + new_brackets = realloc(brackets, + strlen(brackets) + + strlen(token) + len); + if (!new_brackets) { + free(brackets); + goto fail; + } + brackets = new_brackets; + if (len == 2) + strcat(brackets, " "); + strcat(brackets, token); + /* We only care about the last token */ + field->arraylen = strtoul(token, NULL, 0); + free_token(token); + type = read_token(&token); + if (type == EVENT_NONE) { + do_warning("failed to find token"); + goto fail; + } + } + + free_token(token); + + new_brackets = realloc(brackets, strlen(brackets) + 2); + if (!new_brackets) { + free(brackets); + goto fail; + } + brackets = new_brackets; + strcat(brackets, "]"); + + /* add brackets to type */ + + type = read_token(&token); + /* + * If the next token is not an OP, then it is of + * the format: type [] item; + */ + if (type == EVENT_ITEM) { + char *new_type; + new_type = realloc(field->type, + strlen(field->type) + + strlen(field->name) + + strlen(brackets) + 2); + if (!new_type) { + free(brackets); + goto fail; + } + field->type = new_type; + strcat(field->type, " "); + strcat(field->type, field->name); + size_dynamic = type_size(field->name); + free_token(field->name); + strcat(field->type, brackets); + field->name = token; + type = read_token(&token); + } else { + char *new_type; + new_type = realloc(field->type, + strlen(field->type) + + strlen(brackets) + 1); + if (!new_type) { + free(brackets); + goto fail; + } + field->type = new_type; + strcat(field->type, brackets); + } + free(brackets); + } + + if (field_is_string(field)) + field->flags |= FIELD_IS_STRING; + if (field_is_dynamic(field)) + field->flags |= FIELD_IS_DYNAMIC; + if (field_is_long(field)) + field->flags |= FIELD_IS_LONG; + + if (test_type_token(type, token, EVENT_OP, ";")) + goto fail; + free_token(token); + + if (read_expected(EVENT_ITEM, "offset") < 0) + goto fail_expect; + + if (read_expected(EVENT_OP, ":") < 0) + goto fail_expect; + + if (read_expect_type(EVENT_ITEM, &token)) + goto fail; + field->offset = strtoul(token, NULL, 0); + free_token(token); + + if (read_expected(EVENT_OP, ";") < 0) + goto fail_expect; + + if (read_expected(EVENT_ITEM, "size") < 0) + goto fail_expect; + + if (read_expected(EVENT_OP, ":") < 0) + goto fail_expect; + + if (read_expect_type(EVENT_ITEM, &token)) + goto fail; + field->size = strtoul(token, NULL, 0); + free_token(token); + + if (read_expected(EVENT_OP, ";") < 0) + goto fail_expect; + + type = read_token(&token); + if (type != EVENT_NEWLINE) { + /* newer versions of the kernel have a "signed" type */ + if (test_type_token(type, token, EVENT_ITEM, "signed")) + goto fail; + + free_token(token); + + if (read_expected(EVENT_OP, ":") < 0) + goto fail_expect; + + if (read_expect_type(EVENT_ITEM, &token)) + goto fail; + + if (strtoul(token, NULL, 0)) + field->flags |= FIELD_IS_SIGNED; + + free_token(token); + if (read_expected(EVENT_OP, ";") < 0) + goto fail_expect; + + if (read_expect_type(EVENT_NEWLINE, &token)) + goto fail; + } + + free_token(token); + + if (field->flags & FIELD_IS_ARRAY) { + if (field->arraylen) + field->elementsize = field->size / field->arraylen; + else if (field->flags & FIELD_IS_DYNAMIC) + field->elementsize = size_dynamic; + else if (field->flags & FIELD_IS_STRING) + field->elementsize = 1; + else if (field->flags & FIELD_IS_LONG) + field->elementsize = event->pevent ? + event->pevent->long_size : + sizeof(long); + } else + field->elementsize = field->size; + + *fields = field; + fields = &field->next; + + } while (1); + + return 0; + +fail: + free_token(token); +fail_expect: + if (field) { + free(field->type); + free(field->name); + free(field); + } + return -1; +} + +static int event_read_format(struct event_format *event) +{ + char *token; + int ret; + + if (read_expected_item(EVENT_ITEM, "format") < 0) + return -1; + + if (read_expected(EVENT_OP, ":") < 0) + return -1; + + if (read_expect_type(EVENT_NEWLINE, &token)) + goto fail; + free_token(token); + + ret = event_read_fields(event, &event->format.common_fields); + if (ret < 0) + return ret; + event->format.nr_common = ret; + + ret = event_read_fields(event, &event->format.fields); + if (ret < 0) + return ret; + event->format.nr_fields = ret; + + return 0; + + fail: + free_token(token); + return -1; +} + +static enum event_type +process_arg_token(struct event_format *event, struct print_arg *arg, + char **tok, enum event_type type); + +static enum event_type +process_arg(struct event_format *event, struct print_arg *arg, char **tok) +{ + enum event_type type; + char *token; + + type = read_token(&token); + *tok = token; + + return process_arg_token(event, arg, tok, type); +} + +static enum event_type +process_op(struct event_format *event, struct print_arg *arg, char **tok); + +static enum event_type +process_cond(struct event_format *event, struct print_arg *top, char **tok) +{ + struct print_arg *arg, *left, *right; + enum event_type type; + char *token = NULL; + + arg = alloc_arg(); + left = alloc_arg(); + right = alloc_arg(); + + if (!arg || !left || !right) { + do_warning("%s: not enough memory!", __func__); + /* arg will be freed at out_free */ + free_arg(left); + free_arg(right); + goto out_free; + } + + arg->type = PRINT_OP; + arg->op.left = left; + arg->op.right = right; + + *tok = NULL; + type = process_arg(event, left, &token); + + again: + /* Handle other operations in the arguments */ + if (type == EVENT_OP && strcmp(token, ":") != 0) { + type = process_op(event, left, &token); + goto again; + } + + if (test_type_token(type, token, EVENT_OP, ":")) + goto out_free; + + arg->op.op = token; + + type = process_arg(event, right, &token); + + top->op.right = arg; + + *tok = token; + return type; + +out_free: + /* Top may point to itself */ + top->op.right = NULL; + free_token(token); + free_arg(arg); + return EVENT_ERROR; +} + +static enum event_type +process_array(struct event_format *event, struct print_arg *top, char **tok) +{ + struct print_arg *arg; + enum event_type type; + char *token = NULL; + + arg = alloc_arg(); + if (!arg) { + do_warning("%s: not enough memory!", __func__); + /* '*tok' is set to top->op.op. No need to free. */ + *tok = NULL; + return EVENT_ERROR; + } + + *tok = NULL; + type = process_arg(event, arg, &token); + if (test_type_token(type, token, EVENT_OP, "]")) + goto out_free; + + top->op.right = arg; + + free_token(token); + type = read_token_item(&token); + *tok = token; + + return type; + +out_free: + free_token(token); + free_arg(arg); + return EVENT_ERROR; +} + +static int get_op_prio(char *op) +{ + if (!op[1]) { + switch (op[0]) { + case '~': + case '!': + return 4; + case '*': + case '/': + case '%': + return 6; + case '+': + case '-': + return 7; + /* '>>' and '<<' are 8 */ + case '<': + case '>': + return 9; + /* '==' and '!=' are 10 */ + case '&': + return 11; + case '^': + return 12; + case '|': + return 13; + case '?': + return 16; + default: + do_warning("unknown op '%c'", op[0]); + return -1; + } + } else { + if (strcmp(op, "++") == 0 || + strcmp(op, "--") == 0) { + return 3; + } else if (strcmp(op, ">>") == 0 || + strcmp(op, "<<") == 0) { + return 8; + } else if (strcmp(op, ">=") == 0 || + strcmp(op, "<=") == 0) { + return 9; + } else if (strcmp(op, "==") == 0 || + strcmp(op, "!=") == 0) { + return 10; + } else if (strcmp(op, "&&") == 0) { + return 14; + } else if (strcmp(op, "||") == 0) { + return 15; + } else { + do_warning("unknown op '%s'", op); + return -1; + } + } +} + +static int set_op_prio(struct print_arg *arg) +{ + + /* single ops are the greatest */ + if (!arg->op.left || arg->op.left->type == PRINT_NULL) + arg->op.prio = 0; + else + arg->op.prio = get_op_prio(arg->op.op); + + return arg->op.prio; +} + +/* Note, *tok does not get freed, but will most likely be saved */ +static enum event_type +process_op(struct event_format *event, struct print_arg *arg, char **tok) +{ + struct print_arg *left, *right = NULL; + enum event_type type; + char *token; + + /* the op is passed in via tok */ + token = *tok; + + if (arg->type == PRINT_OP && !arg->op.left) { + /* handle single op */ + if (token[1]) { + do_warning("bad op token %s", token); + goto out_free; + } + switch (token[0]) { + case '~': + case '!': + case '+': + case '-': + break; + default: + do_warning("bad op token %s", token); + goto out_free; + + } + + /* make an empty left */ + left = alloc_arg(); + if (!left) + goto out_warn_free; + + left->type = PRINT_NULL; + arg->op.left = left; + + right = alloc_arg(); + if (!right) + goto out_warn_free; + + arg->op.right = right; + + /* do not free the token, it belongs to an op */ + *tok = NULL; + type = process_arg(event, right, tok); + + } else if (strcmp(token, "?") == 0) { + + left = alloc_arg(); + if (!left) + goto out_warn_free; + + /* copy the top arg to the left */ + *left = *arg; + + arg->type = PRINT_OP; + arg->op.op = token; + arg->op.left = left; + arg->op.prio = 0; + + /* it will set arg->op.right */ + type = process_cond(event, arg, tok); + + } else if (strcmp(token, ">>") == 0 || + strcmp(token, "<<") == 0 || + strcmp(token, "&") == 0 || + strcmp(token, "|") == 0 || + strcmp(token, "&&") == 0 || + strcmp(token, "||") == 0 || + strcmp(token, "-") == 0 || + strcmp(token, "+") == 0 || + strcmp(token, "*") == 0 || + strcmp(token, "^") == 0 || + strcmp(token, "/") == 0 || + strcmp(token, "<") == 0 || + strcmp(token, ">") == 0 || + strcmp(token, "<=") == 0 || + strcmp(token, ">=") == 0 || + strcmp(token, "==") == 0 || + strcmp(token, "!=") == 0) { + + left = alloc_arg(); + if (!left) + goto out_warn_free; + + /* copy the top arg to the left */ + *left = *arg; + + arg->type = PRINT_OP; + arg->op.op = token; + arg->op.left = left; + arg->op.right = NULL; + + if (set_op_prio(arg) == -1) { + event->flags |= EVENT_FL_FAILED; + /* arg->op.op (= token) will be freed at out_free */ + arg->op.op = NULL; + goto out_free; + } + + type = read_token_item(&token); + *tok = token; + + /* could just be a type pointer */ + if ((strcmp(arg->op.op, "*") == 0) && + type == EVENT_DELIM && (strcmp(token, ")") == 0)) { + char *new_atom; + + if (left->type != PRINT_ATOM) { + do_warning("bad pointer type"); + goto out_free; + } + new_atom = realloc(left->atom.atom, + strlen(left->atom.atom) + 3); + if (!new_atom) + goto out_warn_free; + + left->atom.atom = new_atom; + strcat(left->atom.atom, " *"); + free(arg->op.op); + *arg = *left; + free(left); + + return type; + } + + right = alloc_arg(); + if (!right) + goto out_warn_free; + + type = process_arg_token(event, right, tok, type); + arg->op.right = right; + + } else if (strcmp(token, "[") == 0) { + + left = alloc_arg(); + if (!left) + goto out_warn_free; + + *left = *arg; + + arg->type = PRINT_OP; + arg->op.op = token; + arg->op.left = left; + + arg->op.prio = 0; + + /* it will set arg->op.right */ + type = process_array(event, arg, tok); + + } else { + do_warning("unknown op '%s'", token); + event->flags |= EVENT_FL_FAILED; + /* the arg is now the left side */ + goto out_free; + } + + if (type == EVENT_OP && strcmp(*tok, ":") != 0) { + int prio; + + /* higher prios need to be closer to the root */ + prio = get_op_prio(*tok); + + if (prio > arg->op.prio) + return process_op(event, arg, tok); + + return process_op(event, right, tok); + } + + return type; + +out_warn_free: + do_warning("%s: not enough memory!", __func__); +out_free: + free_token(token); + *tok = NULL; + return EVENT_ERROR; +} + +static enum event_type +process_entry(struct event_format *event __maybe_unused, struct print_arg *arg, + char **tok) +{ + enum event_type type; + char *field; + char *token; + + if (read_expected(EVENT_OP, "->") < 0) + goto out_err; + + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto out_free; + field = token; + + arg->type = PRINT_FIELD; + arg->field.name = field; + + if (is_flag_field) { + arg->field.field = pevent_find_any_field(event, arg->field.name); + arg->field.field->flags |= FIELD_IS_FLAG; + is_flag_field = 0; + } else if (is_symbolic_field) { + arg->field.field = pevent_find_any_field(event, arg->field.name); + arg->field.field->flags |= FIELD_IS_SYMBOLIC; + is_symbolic_field = 0; + } + + type = read_token(&token); + *tok = token; + + return type; + + out_free: + free_token(token); + out_err: + *tok = NULL; + return EVENT_ERROR; +} + +static char *arg_eval (struct print_arg *arg); + +static unsigned long long +eval_type_str(unsigned long long val, const char *type, int pointer) +{ + int sign = 0; + char *ref; + int len; + + len = strlen(type); + + if (pointer) { + + if (type[len-1] != '*') { + do_warning("pointer expected with non pointer type"); + return val; + } + + ref = malloc(len); + if (!ref) { + do_warning("%s: not enough memory!", __func__); + return val; + } + memcpy(ref, type, len); + + /* chop off the " *" */ + ref[len - 2] = 0; + + val = eval_type_str(val, ref, 0); + free(ref); + return val; + } + + /* check if this is a pointer */ + if (type[len - 1] == '*') + return val; + + /* Try to figure out the arg size*/ + if (strncmp(type, "struct", 6) == 0) + /* all bets off */ + return val; + + if (strcmp(type, "u8") == 0) + return val & 0xff; + + if (strcmp(type, "u16") == 0) + return val & 0xffff; + + if (strcmp(type, "u32") == 0) + return val & 0xffffffff; + + if (strcmp(type, "u64") == 0 || + strcmp(type, "s64")) + return val; + + if (strcmp(type, "s8") == 0) + return (unsigned long long)(char)val & 0xff; + + if (strcmp(type, "s16") == 0) + return (unsigned long long)(short)val & 0xffff; + + if (strcmp(type, "s32") == 0) + return (unsigned long long)(int)val & 0xffffffff; + + if (strncmp(type, "unsigned ", 9) == 0) { + sign = 0; + type += 9; + } + + if (strcmp(type, "char") == 0) { + if (sign) + return (unsigned long long)(char)val & 0xff; + else + return val & 0xff; + } + + if (strcmp(type, "short") == 0) { + if (sign) + return (unsigned long long)(short)val & 0xffff; + else + return val & 0xffff; + } + + if (strcmp(type, "int") == 0) { + if (sign) + return (unsigned long long)(int)val & 0xffffffff; + else + return val & 0xffffffff; + } + + return val; +} + +/* + * Try to figure out the type. + */ +static unsigned long long +eval_type(unsigned long long val, struct print_arg *arg, int pointer) +{ + if (arg->type != PRINT_TYPE) { + do_warning("expected type argument"); + return 0; + } + + return eval_type_str(val, arg->typecast.type, pointer); +} + +static int arg_num_eval(struct print_arg *arg, long long *val) +{ + long long left, right; + int ret = 1; + + switch (arg->type) { + case PRINT_ATOM: + *val = strtoll(arg->atom.atom, NULL, 0); + break; + case PRINT_TYPE: + ret = arg_num_eval(arg->typecast.item, val); + if (!ret) + break; + *val = eval_type(*val, arg, 0); + break; + case PRINT_OP: + switch (arg->op.op[0]) { + case '|': + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + if (arg->op.op[1]) + *val = left || right; + else + *val = left | right; + break; + case '&': + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + if (arg->op.op[1]) + *val = left && right; + else + *val = left & right; + break; + case '<': + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + switch (arg->op.op[1]) { + case 0: + *val = left < right; + break; + case '<': + *val = left << right; + break; + case '=': + *val = left <= right; + break; + default: + do_warning("unknown op '%s'", arg->op.op); + ret = 0; + } + break; + case '>': + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + switch (arg->op.op[1]) { + case 0: + *val = left > right; + break; + case '>': + *val = left >> right; + break; + case '=': + *val = left >= right; + break; + default: + do_warning("unknown op '%s'", arg->op.op); + ret = 0; + } + break; + case '=': + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + + if (arg->op.op[1] != '=') { + do_warning("unknown op '%s'", arg->op.op); + ret = 0; + } else + *val = left == right; + break; + case '!': + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + + switch (arg->op.op[1]) { + case '=': + *val = left != right; + break; + default: + do_warning("unknown op '%s'", arg->op.op); + ret = 0; + } + break; + case '-': + /* check for negative */ + if (arg->op.left->type == PRINT_NULL) + left = 0; + else + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + *val = left - right; + break; + case '+': + if (arg->op.left->type == PRINT_NULL) + left = 0; + else + ret = arg_num_eval(arg->op.left, &left); + if (!ret) + break; + ret = arg_num_eval(arg->op.right, &right); + if (!ret) + break; + *val = left + right; + break; + default: + do_warning("unknown op '%s'", arg->op.op); + ret = 0; + } + break; + + case PRINT_NULL: + case PRINT_FIELD ... PRINT_SYMBOL: + case PRINT_STRING: + case PRINT_BSTRING: + default: + do_warning("invalid eval type %d", arg->type); + ret = 0; + + } + return ret; +} + +static char *arg_eval (struct print_arg *arg) +{ + long long val; + static char buf[20]; + + switch (arg->type) { + case PRINT_ATOM: + return arg->atom.atom; + case PRINT_TYPE: + return arg_eval(arg->typecast.item); + case PRINT_OP: + if (!arg_num_eval(arg, &val)) + break; + sprintf(buf, "%lld", val); + return buf; + + case PRINT_NULL: + case PRINT_FIELD ... PRINT_SYMBOL: + case PRINT_STRING: + case PRINT_BSTRING: + default: + do_warning("invalid eval type %d", arg->type); + break; + } + + return NULL; +} + +static enum event_type +process_fields(struct event_format *event, struct print_flag_sym **list, char **tok) +{ + enum event_type type; + struct print_arg *arg = NULL; + struct print_flag_sym *field; + char *token = *tok; + char *value; + + do { + free_token(token); + type = read_token_item(&token); + if (test_type_token(type, token, EVENT_OP, "{")) + break; + + arg = alloc_arg(); + if (!arg) + goto out_free; + + free_token(token); + type = process_arg(event, arg, &token); + + if (type == EVENT_OP) + type = process_op(event, arg, &token); + + if (type == EVENT_ERROR) + goto out_free; + + if (test_type_token(type, token, EVENT_DELIM, ",")) + goto out_free; + + field = calloc(1, sizeof(*field)); + if (!field) + goto out_free; + + value = arg_eval(arg); + if (value == NULL) + goto out_free_field; + field->value = strdup(value); + if (field->value == NULL) + goto out_free_field; + + free_arg(arg); + arg = alloc_arg(); + if (!arg) + goto out_free; + + free_token(token); + type = process_arg(event, arg, &token); + if (test_type_token(type, token, EVENT_OP, "}")) + goto out_free_field; + + value = arg_eval(arg); + if (value == NULL) + goto out_free_field; + field->str = strdup(value); + if (field->str == NULL) + goto out_free_field; + free_arg(arg); + arg = NULL; + + *list = field; + list = &field->next; + + free_token(token); + type = read_token_item(&token); + } while (type == EVENT_DELIM && strcmp(token, ",") == 0); + + *tok = token; + return type; + +out_free_field: + free_flag_sym(field); +out_free: + free_arg(arg); + free_token(token); + *tok = NULL; + + return EVENT_ERROR; +} + +static enum event_type +process_flags(struct event_format *event, struct print_arg *arg, char **tok) +{ + struct print_arg *field; + enum event_type type; + char *token; + + memset(arg, 0, sizeof(*arg)); + arg->type = PRINT_FLAGS; + + field = alloc_arg(); + if (!field) { + do_warning("%s: not enough memory!", __func__); + goto out_free; + } + + type = process_arg(event, field, &token); + + /* Handle operations in the first argument */ + while (type == EVENT_OP) + type = process_op(event, field, &token); + + if (test_type_token(type, token, EVENT_DELIM, ",")) + goto out_free_field; + free_token(token); + + arg->flags.field = field; + + type = read_token_item(&token); + if (event_item_type(type)) { + arg->flags.delim = token; + type = read_token_item(&token); + } + + if (test_type_token(type, token, EVENT_DELIM, ",")) + goto out_free; + + type = process_fields(event, &arg->flags.flags, &token); + if (test_type_token(type, token, EVENT_DELIM, ")")) + goto out_free; + + free_token(token); + type = read_token_item(tok); + return type; + +out_free_field: + free_arg(field); +out_free: + free_token(token); + *tok = NULL; + return EVENT_ERROR; +} + +static enum event_type +process_symbols(struct event_format *event, struct print_arg *arg, char **tok) +{ + struct print_arg *field; + enum event_type type; + char *token; + + memset(arg, 0, sizeof(*arg)); + arg->type = PRINT_SYMBOL; + + field = alloc_arg(); + if (!field) { + do_warning("%s: not enough memory!", __func__); + goto out_free; + } + + type = process_arg(event, field, &token); + if (test_type_token(type, token, EVENT_DELIM, ",")) + goto out_free_field; + + arg->symbol.field = field; + + type = process_fields(event, &arg->symbol.symbols, &token); + if (test_type_token(type, token, EVENT_DELIM, ")")) + goto out_free; + + free_token(token); + type = read_token_item(tok); + return type; + +out_free_field: + free_arg(field); +out_free: + free_token(token); + *tok = NULL; + return EVENT_ERROR; +} + +static enum event_type +process_hex(struct event_format *event, struct print_arg *arg, char **tok) +{ + struct print_arg *field; + enum event_type type; + char *token; + + memset(arg, 0, sizeof(*arg)); + arg->type = PRINT_HEX; + + field = alloc_arg(); + if (!field) { + do_warning("%s: not enough memory!", __func__); + goto out_free; + } + + type = process_arg(event, field, &token); + + if (test_type_token(type, token, EVENT_DELIM, ",")) + goto out_free; + + arg->hex.field = field; + + free_token(token); + + field = alloc_arg(); + if (!field) { + do_warning("%s: not enough memory!", __func__); + *tok = NULL; + return EVENT_ERROR; + } + + type = process_arg(event, field, &token); + + if (test_type_token(type, token, EVENT_DELIM, ")")) + goto out_free; + + arg->hex.size = field; + + free_token(token); + type = read_token_item(tok); + return type; + + out_free: + free_arg(field); + free_token(token); + *tok = NULL; + return EVENT_ERROR; +} + +static enum event_type +process_dynamic_array(struct event_format *event, struct print_arg *arg, char **tok) +{ + struct format_field *field; + enum event_type type; + char *token; + + memset(arg, 0, sizeof(*arg)); + arg->type = PRINT_DYNAMIC_ARRAY; + + /* + * The item within the parenthesis is another field that holds + * the index into where the array starts. + */ + type = read_token(&token); + *tok = token; + if (type != EVENT_ITEM) + goto out_free; + + /* Find the field */ + + field = pevent_find_field(event, token); + if (!field) + goto out_free; + + arg->dynarray.field = field; + arg->dynarray.index = 0; + + if (read_expected(EVENT_DELIM, ")") < 0) + goto out_free; + + free_token(token); + type = read_token_item(&token); + *tok = token; + if (type != EVENT_OP || strcmp(token, "[") != 0) + return type; + + free_token(token); + arg = alloc_arg(); + if (!arg) { + do_warning("%s: not enough memory!", __func__); + *tok = NULL; + return EVENT_ERROR; + } + + type = process_arg(event, arg, &token); + if (type == EVENT_ERROR) + goto out_free_arg; + + if (!test_type_token(type, token, EVENT_OP, "]")) + goto out_free_arg; + + free_token(token); + type = read_token_item(tok); + return type; + + out_free_arg: + free_arg(arg); + out_free: + free_token(token); + *tok = NULL; + return EVENT_ERROR; +} + +static enum event_type +process_paren(struct event_format *event, struct print_arg *arg, char **tok) +{ + struct print_arg *item_arg; + enum event_type type; + char *token; + + type = process_arg(event, arg, &token); + + if (type == EVENT_ERROR) + goto out_free; + + if (type == EVENT_OP) + type = process_op(event, arg, &token); + + if (type == EVENT_ERROR) + goto out_free; + + if (test_type_token(type, token, EVENT_DELIM, ")")) + goto out_free; + + free_token(token); + type = read_token_item(&token); + + /* + * If the next token is an item or another open paren, then + * this was a typecast. + */ + if (event_item_type(type) || + (type == EVENT_DELIM && strcmp(token, "(") == 0)) { + + /* make this a typecast and contine */ + + /* prevous must be an atom */ + if (arg->type != PRINT_ATOM) { + do_warning("previous needed to be PRINT_ATOM"); + goto out_free; + } + + item_arg = alloc_arg(); + if (!item_arg) { + do_warning("%s: not enough memory!", __func__); + goto out_free; + } + + arg->type = PRINT_TYPE; + arg->typecast.type = arg->atom.atom; + arg->typecast.item = item_arg; + type = process_arg_token(event, item_arg, &token, type); + + } + + *tok = token; + return type; + + out_free: + free_token(token); + *tok = NULL; + return EVENT_ERROR; +} + + +static enum event_type +process_str(struct event_format *event __maybe_unused, struct print_arg *arg, + char **tok) +{ + enum event_type type; + char *token; + + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto out_free; + + arg->type = PRINT_STRING; + arg->string.string = token; + arg->string.offset = -1; + + if (read_expected(EVENT_DELIM, ")") < 0) + goto out_err; + + type = read_token(&token); + *tok = token; + + return type; + + out_free: + free_token(token); + out_err: + *tok = NULL; + return EVENT_ERROR; +} + +static struct pevent_function_handler * +find_func_handler(struct pevent *pevent, char *func_name) +{ + struct pevent_function_handler *func; + + if (!pevent) + return NULL; + + for (func = pevent->func_handlers; func; func = func->next) { + if (strcmp(func->name, func_name) == 0) + break; + } + + return func; +} + +static void remove_func_handler(struct pevent *pevent, char *func_name) +{ + struct pevent_function_handler *func; + struct pevent_function_handler **next; + + next = &pevent->func_handlers; + while ((func = *next)) { + if (strcmp(func->name, func_name) == 0) { + *next = func->next; + free_func_handle(func); + break; + } + next = &func->next; + } +} + +static enum event_type +process_func_handler(struct event_format *event, struct pevent_function_handler *func, + struct print_arg *arg, char **tok) +{ + struct print_arg **next_arg; + struct print_arg *farg; + enum event_type type; + char *token; + const char *test; + int i; + + arg->type = PRINT_FUNC; + arg->func.func = func; + + *tok = NULL; + + next_arg = &(arg->func.args); + for (i = 0; i < func->nr_args; i++) { + farg = alloc_arg(); + if (!farg) { + do_warning("%s: not enough memory!", __func__); + return EVENT_ERROR; + } + + type = process_arg(event, farg, &token); + if (i < (func->nr_args - 1)) + test = ","; + else + test = ")"; + + if (test_type_token(type, token, EVENT_DELIM, test)) { + free_arg(farg); + free_token(token); + return EVENT_ERROR; + } + + *next_arg = farg; + next_arg = &(farg->next); + free_token(token); + } + + type = read_token(&token); + *tok = token; + + return type; +} + +static enum event_type +process_function(struct event_format *event, struct print_arg *arg, + char *token, char **tok) +{ + struct pevent_function_handler *func; + + if (strcmp(token, "__print_flags") == 0) { + free_token(token); + is_flag_field = 1; + return process_flags(event, arg, tok); + } + if (strcmp(token, "__print_symbolic") == 0) { + free_token(token); + is_symbolic_field = 1; + return process_symbols(event, arg, tok); + } + if (strcmp(token, "__print_hex") == 0) { + free_token(token); + return process_hex(event, arg, tok); + } + if (strcmp(token, "__get_str") == 0) { + free_token(token); + return process_str(event, arg, tok); + } + if (strcmp(token, "__get_dynamic_array") == 0) { + free_token(token); + return process_dynamic_array(event, arg, tok); + } + + func = find_func_handler(event->pevent, token); + if (func) { + free_token(token); + return process_func_handler(event, func, arg, tok); + } + + do_warning("function %s not defined", token); + free_token(token); + return EVENT_ERROR; +} + +static enum event_type +process_arg_token(struct event_format *event, struct print_arg *arg, + char **tok, enum event_type type) +{ + char *token; + char *atom; + + token = *tok; + + switch (type) { + case EVENT_ITEM: + if (strcmp(token, "REC") == 0) { + free_token(token); + type = process_entry(event, arg, &token); + break; + } + atom = token; + /* test the next token */ + type = read_token_item(&token); + + /* + * If the next token is a parenthesis, then this + * is a function. + */ + if (type == EVENT_DELIM && strcmp(token, "(") == 0) { + free_token(token); + token = NULL; + /* this will free atom. */ + type = process_function(event, arg, atom, &token); + break; + } + /* atoms can be more than one token long */ + while (type == EVENT_ITEM) { + char *new_atom; + new_atom = realloc(atom, + strlen(atom) + strlen(token) + 2); + if (!new_atom) { + free(atom); + *tok = NULL; + free_token(token); + return EVENT_ERROR; + } + atom = new_atom; + strcat(atom, " "); + strcat(atom, token); + free_token(token); + type = read_token_item(&token); + } + + arg->type = PRINT_ATOM; + arg->atom.atom = atom; + break; + + case EVENT_DQUOTE: + case EVENT_SQUOTE: + arg->type = PRINT_ATOM; + arg->atom.atom = token; + type = read_token_item(&token); + break; + case EVENT_DELIM: + if (strcmp(token, "(") == 0) { + free_token(token); + type = process_paren(event, arg, &token); + break; + } + case EVENT_OP: + /* handle single ops */ + arg->type = PRINT_OP; + arg->op.op = token; + arg->op.left = NULL; + type = process_op(event, arg, &token); + + /* On error, the op is freed */ + if (type == EVENT_ERROR) + arg->op.op = NULL; + + /* return error type if errored */ + break; + + case EVENT_ERROR ... EVENT_NEWLINE: + default: + do_warning("unexpected type %d", type); + return EVENT_ERROR; + } + *tok = token; + + return type; +} + +static int event_read_print_args(struct event_format *event, struct print_arg **list) +{ + enum event_type type = EVENT_ERROR; + struct print_arg *arg; + char *token; + int args = 0; + + do { + if (type == EVENT_NEWLINE) { + type = read_token_item(&token); + continue; + } + + arg = alloc_arg(); + if (!arg) { + do_warning("%s: not enough memory!", __func__); + return -1; + } + + type = process_arg(event, arg, &token); + + if (type == EVENT_ERROR) { + free_token(token); + free_arg(arg); + return -1; + } + + *list = arg; + args++; + + if (type == EVENT_OP) { + type = process_op(event, arg, &token); + free_token(token); + if (type == EVENT_ERROR) { + *list = NULL; + free_arg(arg); + return -1; + } + list = &arg->next; + continue; + } + + if (type == EVENT_DELIM && strcmp(token, ",") == 0) { + free_token(token); + *list = arg; + list = &arg->next; + continue; + } + break; + } while (type != EVENT_NONE); + + if (type != EVENT_NONE && type != EVENT_ERROR) + free_token(token); + + return args; +} + +static int event_read_print(struct event_format *event) +{ + enum event_type type; + char *token; + int ret; + + if (read_expected_item(EVENT_ITEM, "print") < 0) + return -1; + + if (read_expected(EVENT_ITEM, "fmt") < 0) + return -1; + + if (read_expected(EVENT_OP, ":") < 0) + return -1; + + if (read_expect_type(EVENT_DQUOTE, &token) < 0) + goto fail; + + concat: + event->print_fmt.format = token; + event->print_fmt.args = NULL; + + /* ok to have no arg */ + type = read_token_item(&token); + + if (type == EVENT_NONE) + return 0; + + /* Handle concatenation of print lines */ + if (type == EVENT_DQUOTE) { + char *cat; + + if (asprintf(&cat, "%s%s", event->print_fmt.format, token) < 0) + goto fail; + free_token(token); + free_token(event->print_fmt.format); + event->print_fmt.format = NULL; + token = cat; + goto concat; + } + + if (test_type_token(type, token, EVENT_DELIM, ",")) + goto fail; + + free_token(token); + + ret = event_read_print_args(event, &event->print_fmt.args); + if (ret < 0) + return -1; + + return ret; + + fail: + free_token(token); + return -1; +} + +/** + * pevent_find_common_field - return a common field by event + * @event: handle for the event + * @name: the name of the common field to return + * + * Returns a common field from the event by the given @name. + * This only searchs the common fields and not all field. + */ +struct format_field * +pevent_find_common_field(struct event_format *event, const char *name) +{ + struct format_field *format; + + for (format = event->format.common_fields; + format; format = format->next) { + if (strcmp(format->name, name) == 0) + break; + } + + return format; +} + +/** + * pevent_find_field - find a non-common field + * @event: handle for the event + * @name: the name of the non-common field + * + * Returns a non-common field by the given @name. + * This does not search common fields. + */ +struct format_field * +pevent_find_field(struct event_format *event, const char *name) +{ + struct format_field *format; + + for (format = event->format.fields; + format; format = format->next) { + if (strcmp(format->name, name) == 0) + break; + } + + return format; +} + +/** + * pevent_find_any_field - find any field by name + * @event: handle for the event + * @name: the name of the field + * + * Returns a field by the given @name. + * This searchs the common field names first, then + * the non-common ones if a common one was not found. + */ +struct format_field * +pevent_find_any_field(struct event_format *event, const char *name) +{ + struct format_field *format; + + format = pevent_find_common_field(event, name); + if (format) + return format; + return pevent_find_field(event, name); +} + +/** + * pevent_read_number - read a number from data + * @pevent: handle for the pevent + * @ptr: the raw data + * @size: the size of the data that holds the number + * + * Returns the number (converted to host) from the + * raw data. + */ +unsigned long long pevent_read_number(struct pevent *pevent, + const void *ptr, int size) +{ + switch (size) { + case 1: + return *(unsigned char *)ptr; + case 2: + return data2host2(pevent, ptr); + case 4: + return data2host4(pevent, ptr); + case 8: + return data2host8(pevent, ptr); + default: + /* BUG! */ + return 0; + } +} + +/** + * pevent_read_number_field - read a number from data + * @field: a handle to the field + * @data: the raw data to read + * @value: the value to place the number in + * + * Reads raw data according to a field offset and size, + * and translates it into @value. + * + * Returns 0 on success, -1 otherwise. + */ +int pevent_read_number_field(struct format_field *field, const void *data, + unsigned long long *value) +{ + if (!field) + return -1; + switch (field->size) { + case 1: + case 2: + case 4: + case 8: + *value = pevent_read_number(field->event->pevent, + data + field->offset, field->size); + return 0; + default: + return -1; + } +} + +static int get_common_info(struct pevent *pevent, + const char *type, int *offset, int *size) +{ + struct event_format *event; + struct format_field *field; + + /* + * All events should have the same common elements. + * Pick any event to find where the type is; + */ + if (!pevent->events) { + do_warning("no event_list!"); + return -1; + } + + event = pevent->events[0]; + field = pevent_find_common_field(event, type); + if (!field) + return -1; + + *offset = field->offset; + *size = field->size; + + return 0; +} + +static int __parse_common(struct pevent *pevent, void *data, + int *size, int *offset, const char *name) +{ + int ret; + + if (!*size) { + ret = get_common_info(pevent, name, offset, size); + if (ret < 0) + return ret; + } + return pevent_read_number(pevent, data + *offset, *size); +} + +static int trace_parse_common_type(struct pevent *pevent, void *data) +{ + return __parse_common(pevent, data, + &pevent->type_size, &pevent->type_offset, + "common_type"); +} + +static int parse_common_pid(struct pevent *pevent, void *data) +{ + return __parse_common(pevent, data, + &pevent->pid_size, &pevent->pid_offset, + "common_pid"); +} + +static int parse_common_pc(struct pevent *pevent, void *data) +{ + return __parse_common(pevent, data, + &pevent->pc_size, &pevent->pc_offset, + "common_preempt_count"); +} + +static int parse_common_flags(struct pevent *pevent, void *data) +{ + return __parse_common(pevent, data, + &pevent->flags_size, &pevent->flags_offset, + "common_flags"); +} + +static int parse_common_lock_depth(struct pevent *pevent, void *data) +{ + return __parse_common(pevent, data, + &pevent->ld_size, &pevent->ld_offset, + "common_lock_depth"); +} + +static int parse_common_migrate_disable(struct pevent *pevent, void *data) +{ + return __parse_common(pevent, data, + &pevent->ld_size, &pevent->ld_offset, + "common_migrate_disable"); +} + +static int events_id_cmp(const void *a, const void *b); + +/** + * pevent_find_event - find an event by given id + * @pevent: a handle to the pevent + * @id: the id of the event + * + * Returns an event that has a given @id. + */ +struct event_format *pevent_find_event(struct pevent *pevent, int id) +{ + struct event_format **eventptr; + struct event_format key; + struct event_format *pkey = &key; + + /* Check cache first */ + if (pevent->last_event && pevent->last_event->id == id) + return pevent->last_event; + + key.id = id; + + eventptr = bsearch(&pkey, pevent->events, pevent->nr_events, + sizeof(*pevent->events), events_id_cmp); + + if (eventptr) { + pevent->last_event = *eventptr; + return *eventptr; + } + + return NULL; +} + +/** + * pevent_find_event_by_name - find an event by given name + * @pevent: a handle to the pevent + * @sys: the system name to search for + * @name: the name of the event to search for + * + * This returns an event with a given @name and under the system + * @sys. If @sys is NULL the first event with @name is returned. + */ +struct event_format * +pevent_find_event_by_name(struct pevent *pevent, + const char *sys, const char *name) +{ + struct event_format *event; + int i; + + if (pevent->last_event && + strcmp(pevent->last_event->name, name) == 0 && + (!sys || strcmp(pevent->last_event->system, sys) == 0)) + return pevent->last_event; + + for (i = 0; i < pevent->nr_events; i++) { + event = pevent->events[i]; + if (strcmp(event->name, name) == 0) { + if (!sys) + break; + if (strcmp(event->system, sys) == 0) + break; + } + } + if (i == pevent->nr_events) + event = NULL; + + pevent->last_event = event; + return event; +} + +static unsigned long long +eval_num_arg(void *data, int size, struct event_format *event, struct print_arg *arg) +{ + struct pevent *pevent = event->pevent; + unsigned long long val = 0; + unsigned long long left, right; + struct print_arg *typearg = NULL; + struct print_arg *larg; + unsigned long offset; + unsigned int field_size; + + switch (arg->type) { + case PRINT_NULL: + /* ?? */ + return 0; + case PRINT_ATOM: + return strtoull(arg->atom.atom, NULL, 0); + case PRINT_FIELD: + if (!arg->field.field) { + arg->field.field = pevent_find_any_field(event, arg->field.name); + if (!arg->field.field) + goto out_warning_field; + + } + /* must be a number */ + val = pevent_read_number(pevent, data + arg->field.field->offset, + arg->field.field->size); + break; + case PRINT_FLAGS: + case PRINT_SYMBOL: + case PRINT_HEX: + break; + case PRINT_TYPE: + val = eval_num_arg(data, size, event, arg->typecast.item); + return eval_type(val, arg, 0); + case PRINT_STRING: + case PRINT_BSTRING: + return 0; + case PRINT_FUNC: { + struct trace_seq s; + trace_seq_init(&s); + val = process_defined_func(&s, data, size, event, arg); + trace_seq_destroy(&s); + return val; + } + case PRINT_OP: + if (strcmp(arg->op.op, "[") == 0) { + /* + * Arrays are special, since we don't want + * to read the arg as is. + */ + right = eval_num_arg(data, size, event, arg->op.right); + + /* handle typecasts */ + larg = arg->op.left; + while (larg->type == PRINT_TYPE) { + if (!typearg) + typearg = larg; + larg = larg->typecast.item; + } + + /* Default to long size */ + field_size = pevent->long_size; + + switch (larg->type) { + case PRINT_DYNAMIC_ARRAY: + offset = pevent_read_number(pevent, + data + larg->dynarray.field->offset, + larg->dynarray.field->size); + if (larg->dynarray.field->elementsize) + field_size = larg->dynarray.field->elementsize; + /* + * The actual length of the dynamic array is stored + * in the top half of the field, and the offset + * is in the bottom half of the 32 bit field. + */ + offset &= 0xffff; + offset += right; + break; + case PRINT_FIELD: + if (!larg->field.field) { + larg->field.field = + pevent_find_any_field(event, larg->field.name); + if (!larg->field.field) { + arg = larg; + goto out_warning_field; + } + } + field_size = larg->field.field->elementsize; + offset = larg->field.field->offset + + right * larg->field.field->elementsize; + break; + default: + goto default_op; /* oops, all bets off */ + } + val = pevent_read_number(pevent, + data + offset, field_size); + if (typearg) + val = eval_type(val, typearg, 1); + break; + } else if (strcmp(arg->op.op, "?") == 0) { + left = eval_num_arg(data, size, event, arg->op.left); + arg = arg->op.right; + if (left) + val = eval_num_arg(data, size, event, arg->op.left); + else + val = eval_num_arg(data, size, event, arg->op.right); + break; + } + default_op: + left = eval_num_arg(data, size, event, arg->op.left); + right = eval_num_arg(data, size, event, arg->op.right); + switch (arg->op.op[0]) { + case '!': + switch (arg->op.op[1]) { + case 0: + val = !right; + break; + case '=': + val = left != right; + break; + default: + goto out_warning_op; + } + break; + case '~': + val = ~right; + break; + case '|': + if (arg->op.op[1]) + val = left || right; + else + val = left | right; + break; + case '&': + if (arg->op.op[1]) + val = left && right; + else + val = left & right; + break; + case '<': + switch (arg->op.op[1]) { + case 0: + val = left < right; + break; + case '<': + val = left << right; + break; + case '=': + val = left <= right; + break; + default: + goto out_warning_op; + } + break; + case '>': + switch (arg->op.op[1]) { + case 0: + val = left > right; + break; + case '>': + val = left >> right; + break; + case '=': + val = left >= right; + break; + default: + goto out_warning_op; + } + break; + case '=': + if (arg->op.op[1] != '=') + goto out_warning_op; + + val = left == right; + break; + case '-': + val = left - right; + break; + case '+': + val = left + right; + break; + case '/': + val = left / right; + break; + case '*': + val = left * right; + break; + default: + goto out_warning_op; + } + break; + default: /* not sure what to do there */ + return 0; + } + return val; + +out_warning_op: + do_warning("%s: unknown op '%s'", __func__, arg->op.op); + return 0; + +out_warning_field: + do_warning("%s: field %s not found", __func__, arg->field.name); + return 0; +} + +struct flag { + const char *name; + unsigned long long value; +}; + +static const struct flag flags[] = { + { "HI_SOFTIRQ", 0 }, + { "TIMER_SOFTIRQ", 1 }, + { "NET_TX_SOFTIRQ", 2 }, + { "NET_RX_SOFTIRQ", 3 }, + { "BLOCK_SOFTIRQ", 4 }, + { "BLOCK_IOPOLL_SOFTIRQ", 5 }, + { "TASKLET_SOFTIRQ", 6 }, + { "SCHED_SOFTIRQ", 7 }, + { "HRTIMER_SOFTIRQ", 8 }, + { "RCU_SOFTIRQ", 9 }, + + { "HRTIMER_NORESTART", 0 }, + { "HRTIMER_RESTART", 1 }, +}; + +static unsigned long long eval_flag(const char *flag) +{ + int i; + + /* + * Some flags in the format files do not get converted. + * If the flag is not numeric, see if it is something that + * we already know about. + */ + if (isdigit(flag[0])) + return strtoull(flag, NULL, 0); + + for (i = 0; i < (int)(sizeof(flags)/sizeof(flags[0])); i++) + if (strcmp(flags[i].name, flag) == 0) + return flags[i].value; + + return 0; +} + +static void print_str_to_seq(struct trace_seq *s, const char *format, + int len_arg, const char *str) +{ + if (len_arg >= 0) + trace_seq_printf(s, format, len_arg, str); + else + trace_seq_printf(s, format, str); +} + +static void print_str_arg(struct trace_seq *s, void *data, int size, + struct event_format *event, const char *format, + int len_arg, struct print_arg *arg) +{ + struct pevent *pevent = event->pevent; + struct print_flag_sym *flag; + struct format_field *field; + unsigned long long val, fval; + unsigned long addr; + char *str; + unsigned char *hex; + int print; + int i, len; + + switch (arg->type) { + case PRINT_NULL: + /* ?? */ + return; + case PRINT_ATOM: + print_str_to_seq(s, format, len_arg, arg->atom.atom); + return; + case PRINT_FIELD: + field = arg->field.field; + if (!field) { + field = pevent_find_any_field(event, arg->field.name); + if (!field) { + str = arg->field.name; + goto out_warning_field; + } + arg->field.field = field; + } + /* Zero sized fields, mean the rest of the data */ + len = field->size ? : size - field->offset; + + /* + * Some events pass in pointers. If this is not an array + * and the size is the same as long_size, assume that it + * is a pointer. + */ + if (!(field->flags & FIELD_IS_ARRAY) && + field->size == pevent->long_size) { + addr = *(unsigned long *)(data + field->offset); + trace_seq_printf(s, "%lx", addr); + break; + } + str = malloc(len + 1); + if (!str) { + do_warning("%s: not enough memory!", __func__); + return; + } + memcpy(str, data + field->offset, len); + str[len] = 0; + print_str_to_seq(s, format, len_arg, str); + free(str); + break; + case PRINT_FLAGS: + val = eval_num_arg(data, size, event, arg->flags.field); + print = 0; + for (flag = arg->flags.flags; flag; flag = flag->next) { + fval = eval_flag(flag->value); + if (!val && !fval) { + print_str_to_seq(s, format, len_arg, flag->str); + break; + } + if (fval && (val & fval) == fval) { + if (print && arg->flags.delim) + trace_seq_puts(s, arg->flags.delim); + print_str_to_seq(s, format, len_arg, flag->str); + print = 1; + val &= ~fval; + } + } + break; + case PRINT_SYMBOL: + val = eval_num_arg(data, size, event, arg->symbol.field); + for (flag = arg->symbol.symbols; flag; flag = flag->next) { + fval = eval_flag(flag->value); + if (val == fval) { + print_str_to_seq(s, format, len_arg, flag->str); + break; + } + } + break; + case PRINT_HEX: + field = arg->hex.field->field.field; + if (!field) { + str = arg->hex.field->field.name; + field = pevent_find_any_field(event, str); + if (!field) + goto out_warning_field; + arg->hex.field->field.field = field; + } + hex = data + field->offset; + len = eval_num_arg(data, size, event, arg->hex.size); + for (i = 0; i < len; i++) { + if (i) + trace_seq_putc(s, ' '); + trace_seq_printf(s, "%02x", hex[i]); + } + break; + + case PRINT_TYPE: + break; + case PRINT_STRING: { + int str_offset; + + if (arg->string.offset == -1) { + struct format_field *f; + + f = pevent_find_any_field(event, arg->string.string); + arg->string.offset = f->offset; + } + str_offset = data2host4(pevent, data + arg->string.offset); + str_offset &= 0xffff; + print_str_to_seq(s, format, len_arg, ((char *)data) + str_offset); + break; + } + case PRINT_BSTRING: + print_str_to_seq(s, format, len_arg, arg->string.string); + break; + case PRINT_OP: + /* + * The only op for string should be ? : + */ + if (arg->op.op[0] != '?') + return; + val = eval_num_arg(data, size, event, arg->op.left); + if (val) + print_str_arg(s, data, size, event, + format, len_arg, arg->op.right->op.left); + else + print_str_arg(s, data, size, event, + format, len_arg, arg->op.right->op.right); + break; + case PRINT_FUNC: + process_defined_func(s, data, size, event, arg); + break; + default: + /* well... */ + break; + } + + return; + +out_warning_field: + do_warning("%s: field %s not found", __func__, arg->field.name); +} + +static unsigned long long +process_defined_func(struct trace_seq *s, void *data, int size, + struct event_format *event, struct print_arg *arg) +{ + struct pevent_function_handler *func_handle = arg->func.func; + struct pevent_func_params *param; + unsigned long long *args; + unsigned long long ret; + struct print_arg *farg; + struct trace_seq str; + struct save_str { + struct save_str *next; + char *str; + } *strings = NULL, *string; + int i; + + if (!func_handle->nr_args) { + ret = (*func_handle->func)(s, NULL); + goto out; + } + + farg = arg->func.args; + param = func_handle->params; + + ret = ULLONG_MAX; + args = malloc(sizeof(*args) * func_handle->nr_args); + if (!args) + goto out; + + for (i = 0; i < func_handle->nr_args; i++) { + switch (param->type) { + case PEVENT_FUNC_ARG_INT: + case PEVENT_FUNC_ARG_LONG: + case PEVENT_FUNC_ARG_PTR: + args[i] = eval_num_arg(data, size, event, farg); + break; + case PEVENT_FUNC_ARG_STRING: + trace_seq_init(&str); + print_str_arg(&str, data, size, event, "%s", -1, farg); + trace_seq_terminate(&str); + string = malloc(sizeof(*string)); + if (!string) { + do_warning("%s(%d): malloc str", __func__, __LINE__); + goto out_free; + } + string->next = strings; + string->str = strdup(str.buffer); + if (!string->str) { + free(string); + do_warning("%s(%d): malloc str", __func__, __LINE__); + goto out_free; + } + args[i] = (uintptr_t)string->str; + strings = string; + trace_seq_destroy(&str); + break; + default: + /* + * Something went totally wrong, this is not + * an input error, something in this code broke. + */ + do_warning("Unexpected end of arguments\n"); + goto out_free; + } + farg = farg->next; + param = param->next; + } + + ret = (*func_handle->func)(s, args); +out_free: + free(args); + while (strings) { + string = strings; + strings = string->next; + free(string->str); + free(string); + } + + out: + /* TBD : handle return type here */ + return ret; +} + +static void free_args(struct print_arg *args) +{ + struct print_arg *next; + + while (args) { + next = args->next; + + free_arg(args); + args = next; + } +} + +static struct print_arg *make_bprint_args(char *fmt, void *data, int size, struct event_format *event) +{ + struct pevent *pevent = event->pevent; + struct format_field *field, *ip_field; + struct print_arg *args, *arg, **next; + unsigned long long ip, val; + char *ptr; + void *bptr; + int vsize; + + field = pevent->bprint_buf_field; + ip_field = pevent->bprint_ip_field; + + if (!field) { + field = pevent_find_field(event, "buf"); + if (!field) { + do_warning("can't find buffer field for binary printk"); + return NULL; + } + ip_field = pevent_find_field(event, "ip"); + if (!ip_field) { + do_warning("can't find ip field for binary printk"); + return NULL; + } + pevent->bprint_buf_field = field; + pevent->bprint_ip_field = ip_field; + } + + ip = pevent_read_number(pevent, data + ip_field->offset, ip_field->size); + + /* + * The first arg is the IP pointer. + */ + args = alloc_arg(); + if (!args) { + do_warning("%s(%d): not enough memory!", __func__, __LINE__); + return NULL; + } + arg = args; + arg->next = NULL; + next = &arg->next; + + arg->type = PRINT_ATOM; + + if (asprintf(&arg->atom.atom, "%lld", ip) < 0) + goto out_free; + + /* skip the first "%pf : " */ + for (ptr = fmt + 6, bptr = data + field->offset; + bptr < data + size && *ptr; ptr++) { + int ls = 0; + + if (*ptr == '%') { + process_again: + ptr++; + switch (*ptr) { + case '%': + break; + case 'l': + ls++; + goto process_again; + case 'L': + ls = 2; + goto process_again; + case '0' ... '9': + goto process_again; + case '.': + goto process_again; + case 'p': + ls = 1; + /* fall through */ + case 'd': + case 'u': + case 'x': + case 'i': + switch (ls) { + case 0: + vsize = 4; + break; + case 1: + vsize = pevent->long_size; + break; + case 2: + vsize = 8; + break; + default: + vsize = ls; /* ? */ + break; + } + /* fall through */ + case '*': + if (*ptr == '*') + vsize = 4; + + /* the pointers are always 4 bytes aligned */ + bptr = (void *)(((unsigned long)bptr + 3) & + ~3); + val = pevent_read_number(pevent, bptr, vsize); + bptr += vsize; + arg = alloc_arg(); + if (!arg) { + do_warning("%s(%d): not enough memory!", + __func__, __LINE__); + goto out_free; + } + arg->next = NULL; + arg->type = PRINT_ATOM; + if (asprintf(&arg->atom.atom, "%lld", val) < 0) { + free(arg); + goto out_free; + } + *next = arg; + next = &arg->next; + /* + * The '*' case means that an arg is used as the length. + * We need to continue to figure out for what. + */ + if (*ptr == '*') + goto process_again; + + break; + case 's': + arg = alloc_arg(); + if (!arg) { + do_warning("%s(%d): not enough memory!", + __func__, __LINE__); + goto out_free; + } + arg->next = NULL; + arg->type = PRINT_BSTRING; + arg->string.string = strdup(bptr); + if (!arg->string.string) + goto out_free; + bptr += strlen(bptr) + 1; + *next = arg; + next = &arg->next; + default: + break; + } + } + } + + return args; + +out_free: + free_args(args); + return NULL; +} + +static char * +get_bprint_format(void *data, int size __maybe_unused, + struct event_format *event) +{ + struct pevent *pevent = event->pevent; + unsigned long long addr; + struct format_field *field; + struct printk_map *printk; + char *format; + char *p; + + field = pevent->bprint_fmt_field; + + if (!field) { + field = pevent_find_field(event, "fmt"); + if (!field) { + do_warning("can't find format field for binary printk"); + return NULL; + } + pevent->bprint_fmt_field = field; + } + + addr = pevent_read_number(pevent, data + field->offset, field->size); + + printk = find_printk(pevent, addr); + if (!printk) { + if (asprintf(&format, "%%pf : (NO FORMAT FOUND at %llx)\n", addr) < 0) + return NULL; + return format; + } + + p = printk->printk; + /* Remove any quotes. */ + if (*p == '"') + p++; + if (asprintf(&format, "%s : %s", "%pf", p) < 0) + return NULL; + /* remove ending quotes and new line since we will add one too */ + p = format + strlen(format) - 1; + if (*p == '"') + *p = 0; + + p -= 2; + if (strcmp(p, "\\n") == 0) + *p = 0; + + return format; +} + +static void print_mac_arg(struct trace_seq *s, int mac, void *data, int size, + struct event_format *event, struct print_arg *arg) +{ + unsigned char *buf; + const char *fmt = "%.2x:%.2x:%.2x:%.2x:%.2x:%.2x"; + + if (arg->type == PRINT_FUNC) { + process_defined_func(s, data, size, event, arg); + return; + } + + if (arg->type != PRINT_FIELD) { + trace_seq_printf(s, "ARG TYPE NOT FIELD BUT %d", + arg->type); + return; + } + + if (mac == 'm') + fmt = "%.2x%.2x%.2x%.2x%.2x%.2x"; + if (!arg->field.field) { + arg->field.field = + pevent_find_any_field(event, arg->field.name); + if (!arg->field.field) { + do_warning("%s: field %s not found", + __func__, arg->field.name); + return; + } + } + if (arg->field.field->size != 6) { + trace_seq_printf(s, "INVALIDMAC"); + return; + } + buf = data + arg->field.field->offset; + trace_seq_printf(s, fmt, buf[0], buf[1], buf[2], buf[3], buf[4], buf[5]); +} + +static int is_printable_array(char *p, unsigned int len) +{ + unsigned int i; + + for (i = 0; i < len && p[i]; i++) + if (!isprint(p[i])) + return 0; + return 1; +} + +static void print_event_fields(struct trace_seq *s, void *data, + int size __maybe_unused, + struct event_format *event) +{ + struct format_field *field; + unsigned long long val; + unsigned int offset, len, i; + + field = event->format.fields; + while (field) { + trace_seq_printf(s, " %s=", field->name); + if (field->flags & FIELD_IS_ARRAY) { + offset = field->offset; + len = field->size; + if (field->flags & FIELD_IS_DYNAMIC) { + val = pevent_read_number(event->pevent, data + offset, len); + offset = val; + len = offset >> 16; + offset &= 0xffff; + } + if (field->flags & FIELD_IS_STRING && + is_printable_array(data + offset, len)) { + trace_seq_printf(s, "%s", (char *)data + offset); + } else { + trace_seq_puts(s, "ARRAY["); + for (i = 0; i < len; i++) { + if (i) + trace_seq_puts(s, ", "); + trace_seq_printf(s, "%02x", + *((unsigned char *)data + offset + i)); + } + trace_seq_putc(s, ']'); + field->flags &= ~FIELD_IS_STRING; + } + } else { + val = pevent_read_number(event->pevent, data + field->offset, + field->size); + if (field->flags & FIELD_IS_POINTER) { + trace_seq_printf(s, "0x%llx", val); + } else if (field->flags & FIELD_IS_SIGNED) { + switch (field->size) { + case 4: + /* + * If field is long then print it in hex. + * A long usually stores pointers. + */ + if (field->flags & FIELD_IS_LONG) + trace_seq_printf(s, "0x%x", (int)val); + else + trace_seq_printf(s, "%d", (int)val); + break; + case 2: + trace_seq_printf(s, "%2d", (short)val); + break; + case 1: + trace_seq_printf(s, "%1d", (char)val); + break; + default: + trace_seq_printf(s, "%lld", val); + } + } else { + if (field->flags & FIELD_IS_LONG) + trace_seq_printf(s, "0x%llx", val); + else + trace_seq_printf(s, "%llu", val); + } + } + field = field->next; + } +} + +static void pretty_print(struct trace_seq *s, void *data, int size, struct event_format *event) +{ + struct pevent *pevent = event->pevent; + struct print_fmt *print_fmt = &event->print_fmt; + struct print_arg *arg = print_fmt->args; + struct print_arg *args = NULL; + const char *ptr = print_fmt->format; + unsigned long long val; + struct func_map *func; + const char *saveptr; + char *bprint_fmt = NULL; + char format[32]; + int show_func; + int len_as_arg; + int len_arg; + int len; + int ls; + + if (event->flags & EVENT_FL_FAILED) { + trace_seq_printf(s, "[FAILED TO PARSE]"); + print_event_fields(s, data, size, event); + return; + } + + if (event->flags & EVENT_FL_ISBPRINT) { + bprint_fmt = get_bprint_format(data, size, event); + args = make_bprint_args(bprint_fmt, data, size, event); + arg = args; + ptr = bprint_fmt; + } + + for (; *ptr; ptr++) { + ls = 0; + if (*ptr == '\\') { + ptr++; + switch (*ptr) { + case 'n': + trace_seq_putc(s, '\n'); + break; + case 't': + trace_seq_putc(s, '\t'); + break; + case 'r': + trace_seq_putc(s, '\r'); + break; + case '\\': + trace_seq_putc(s, '\\'); + break; + default: + trace_seq_putc(s, *ptr); + break; + } + + } else if (*ptr == '%') { + saveptr = ptr; + show_func = 0; + len_as_arg = 0; + cont_process: + ptr++; + switch (*ptr) { + case '%': + trace_seq_putc(s, '%'); + break; + case '#': + /* FIXME: need to handle properly */ + goto cont_process; + case 'h': + ls--; + goto cont_process; + case 'l': + ls++; + goto cont_process; + case 'L': + ls = 2; + goto cont_process; + case '*': + /* The argument is the length. */ + if (!arg) { + do_warning("no argument match"); + event->flags |= EVENT_FL_FAILED; + goto out_failed; + } + len_arg = eval_num_arg(data, size, event, arg); + len_as_arg = 1; + arg = arg->next; + goto cont_process; + case '.': + case 'z': + case 'Z': + case '0' ... '9': + goto cont_process; + case 'p': + if (pevent->long_size == 4) + ls = 1; + else + ls = 2; + + if (*(ptr+1) == 'F' || + *(ptr+1) == 'f') { + ptr++; + show_func = *ptr; + } else if (*(ptr+1) == 'M' || *(ptr+1) == 'm') { + print_mac_arg(s, *(ptr+1), data, size, event, arg); + ptr++; + arg = arg->next; + break; + } + + /* fall through */ + case 'd': + case 'i': + case 'x': + case 'X': + case 'u': + if (!arg) { + do_warning("no argument match"); + event->flags |= EVENT_FL_FAILED; + goto out_failed; + } + + len = ((unsigned long)ptr + 1) - + (unsigned long)saveptr; + + /* should never happen */ + if (len > 31) { + do_warning("bad format!"); + event->flags |= EVENT_FL_FAILED; + len = 31; + } + + memcpy(format, saveptr, len); + format[len] = 0; + + val = eval_num_arg(data, size, event, arg); + arg = arg->next; + + if (show_func) { + func = find_func(pevent, val); + if (func) { + trace_seq_puts(s, func->func); + if (show_func == 'F') + trace_seq_printf(s, + "+0x%llx", + val - func->addr); + break; + } + } + if (pevent->long_size == 8 && ls && + sizeof(long) != 8) { + char *p; + + ls = 2; + /* make %l into %ll */ + p = strchr(format, 'l'); + if (p) + memmove(p+1, p, strlen(p)+1); + else if (strcmp(format, "%p") == 0) + strcpy(format, "0x%llx"); + } + switch (ls) { + case -2: + if (len_as_arg) + trace_seq_printf(s, format, len_arg, (char)val); + else + trace_seq_printf(s, format, (char)val); + break; + case -1: + if (len_as_arg) + trace_seq_printf(s, format, len_arg, (short)val); + else + trace_seq_printf(s, format, (short)val); + break; + case 0: + if (len_as_arg) + trace_seq_printf(s, format, len_arg, (int)val); + else + trace_seq_printf(s, format, (int)val); + break; + case 1: + if (len_as_arg) + trace_seq_printf(s, format, len_arg, (long)val); + else + trace_seq_printf(s, format, (long)val); + break; + case 2: + if (len_as_arg) + trace_seq_printf(s, format, len_arg, + (long long)val); + else + trace_seq_printf(s, format, (long long)val); + break; + default: + do_warning("bad count (%d)", ls); + event->flags |= EVENT_FL_FAILED; + } + break; + case 's': + if (!arg) { + do_warning("no matching argument"); + event->flags |= EVENT_FL_FAILED; + goto out_failed; + } + + len = ((unsigned long)ptr + 1) - + (unsigned long)saveptr; + + /* should never happen */ + if (len > 31) { + do_warning("bad format!"); + event->flags |= EVENT_FL_FAILED; + len = 31; + } + + memcpy(format, saveptr, len); + format[len] = 0; + if (!len_as_arg) + len_arg = -1; + print_str_arg(s, data, size, event, + format, len_arg, arg); + arg = arg->next; + break; + default: + trace_seq_printf(s, ">%c<", *ptr); + + } + } else + trace_seq_putc(s, *ptr); + } + + if (event->flags & EVENT_FL_FAILED) { +out_failed: + trace_seq_printf(s, "[FAILED TO PARSE]"); + } + + if (args) { + free_args(args); + free(bprint_fmt); + } +} + +/** + * pevent_data_lat_fmt - parse the data for the latency format + * @pevent: a handle to the pevent + * @s: the trace_seq to write to + * @record: the record to read from + * + * This parses out the Latency format (interrupts disabled, + * need rescheduling, in hard/soft interrupt, preempt count + * and lock depth) and places it into the trace_seq. + */ +void pevent_data_lat_fmt(struct pevent *pevent, + struct trace_seq *s, struct pevent_record *record) +{ + static int check_lock_depth = 1; + static int check_migrate_disable = 1; + static int lock_depth_exists; + static int migrate_disable_exists; + unsigned int lat_flags; + unsigned int pc; + int lock_depth; + int migrate_disable; + int hardirq; + int softirq; + void *data = record->data; + + lat_flags = parse_common_flags(pevent, data); + pc = parse_common_pc(pevent, data); + /* lock_depth may not always exist */ + if (lock_depth_exists) + lock_depth = parse_common_lock_depth(pevent, data); + else if (check_lock_depth) { + lock_depth = parse_common_lock_depth(pevent, data); + if (lock_depth < 0) + check_lock_depth = 0; + else + lock_depth_exists = 1; + } + + /* migrate_disable may not always exist */ + if (migrate_disable_exists) + migrate_disable = parse_common_migrate_disable(pevent, data); + else if (check_migrate_disable) { + migrate_disable = parse_common_migrate_disable(pevent, data); + if (migrate_disable < 0) + check_migrate_disable = 0; + else + migrate_disable_exists = 1; + } + + hardirq = lat_flags & TRACE_FLAG_HARDIRQ; + softirq = lat_flags & TRACE_FLAG_SOFTIRQ; + + trace_seq_printf(s, "%c%c%c", + (lat_flags & TRACE_FLAG_IRQS_OFF) ? 'd' : + (lat_flags & TRACE_FLAG_IRQS_NOSUPPORT) ? + 'X' : '.', + (lat_flags & TRACE_FLAG_NEED_RESCHED) ? + 'N' : '.', + (hardirq && softirq) ? 'H' : + hardirq ? 'h' : softirq ? 's' : '.'); + + if (pc) + trace_seq_printf(s, "%x", pc); + else + trace_seq_putc(s, '.'); + + if (migrate_disable_exists) { + if (migrate_disable < 0) + trace_seq_putc(s, '.'); + else + trace_seq_printf(s, "%d", migrate_disable); + } + + if (lock_depth_exists) { + if (lock_depth < 0) + trace_seq_putc(s, '.'); + else + trace_seq_printf(s, "%d", lock_depth); + } + + trace_seq_terminate(s); +} + +/** + * pevent_data_type - parse out the given event type + * @pevent: a handle to the pevent + * @rec: the record to read from + * + * This returns the event id from the @rec. + */ +int pevent_data_type(struct pevent *pevent, struct pevent_record *rec) +{ + return trace_parse_common_type(pevent, rec->data); +} + +/** + * pevent_data_event_from_type - find the event by a given type + * @pevent: a handle to the pevent + * @type: the type of the event. + * + * This returns the event form a given @type; + */ +struct event_format *pevent_data_event_from_type(struct pevent *pevent, int type) +{ + return pevent_find_event(pevent, type); +} + +/** + * pevent_data_pid - parse the PID from raw data + * @pevent: a handle to the pevent + * @rec: the record to parse + * + * This returns the PID from a raw data. + */ +int pevent_data_pid(struct pevent *pevent, struct pevent_record *rec) +{ + return parse_common_pid(pevent, rec->data); +} + +/** + * pevent_data_comm_from_pid - return the command line from PID + * @pevent: a handle to the pevent + * @pid: the PID of the task to search for + * + * This returns a pointer to the command line that has the given + * @pid. + */ +const char *pevent_data_comm_from_pid(struct pevent *pevent, int pid) +{ + const char *comm; + + comm = find_cmdline(pevent, pid); + return comm; +} + +/** + * pevent_data_comm_from_pid - parse the data into the print format + * @s: the trace_seq to write to + * @event: the handle to the event + * @record: the record to read from + * + * This parses the raw @data using the given @event information and + * writes the print format into the trace_seq. + */ +void pevent_event_info(struct trace_seq *s, struct event_format *event, + struct pevent_record *record) +{ + int print_pretty = 1; + + if (event->pevent->print_raw) + print_event_fields(s, record->data, record->size, event); + else { + + if (event->handler) + print_pretty = event->handler(s, record, event, + event->context); + + if (print_pretty) + pretty_print(s, record->data, record->size, event); + } + + trace_seq_terminate(s); +} + +void pevent_print_event(struct pevent *pevent, struct trace_seq *s, + struct pevent_record *record) +{ + static const char *spaces = " "; /* 20 spaces */ + struct event_format *event; + unsigned long secs; + unsigned long usecs; + unsigned long nsecs; + const char *comm; + void *data = record->data; + int type; + int pid; + int len; + int p; + + secs = record->ts / NSECS_PER_SEC; + nsecs = record->ts - secs * NSECS_PER_SEC; + + if (record->size < 0) { + do_warning("ug! negative record size %d", record->size); + return; + } + + type = trace_parse_common_type(pevent, data); + + event = pevent_find_event(pevent, type); + if (!event) { + do_warning("ug! no event found for type %d", type); + return; + } + + pid = parse_common_pid(pevent, data); + comm = find_cmdline(pevent, pid); + + if (pevent->latency_format) { + trace_seq_printf(s, "%8.8s-%-5d %3d", + comm, pid, record->cpu); + pevent_data_lat_fmt(pevent, s, record); + } else + trace_seq_printf(s, "%16s-%-5d [%03d]", comm, pid, record->cpu); + + if (pevent->flags & PEVENT_NSEC_OUTPUT) { + usecs = nsecs; + p = 9; + } else { + usecs = (nsecs + 500) / NSECS_PER_USEC; + p = 6; + } + + trace_seq_printf(s, " %5lu.%0*lu: %s: ", secs, p, usecs, event->name); + + /* Space out the event names evenly. */ + len = strlen(event->name); + if (len < 20) + trace_seq_printf(s, "%.*s", 20 - len, spaces); + + pevent_event_info(s, event, record); +} + +static int events_id_cmp(const void *a, const void *b) +{ + struct event_format * const * ea = a; + struct event_format * const * eb = b; + + if ((*ea)->id < (*eb)->id) + return -1; + + if ((*ea)->id > (*eb)->id) + return 1; + + return 0; +} + +static int events_name_cmp(const void *a, const void *b) +{ + struct event_format * const * ea = a; + struct event_format * const * eb = b; + int res; + + res = strcmp((*ea)->name, (*eb)->name); + if (res) + return res; + + res = strcmp((*ea)->system, (*eb)->system); + if (res) + return res; + + return events_id_cmp(a, b); +} + +static int events_system_cmp(const void *a, const void *b) +{ + struct event_format * const * ea = a; + struct event_format * const * eb = b; + int res; + + res = strcmp((*ea)->system, (*eb)->system); + if (res) + return res; + + res = strcmp((*ea)->name, (*eb)->name); + if (res) + return res; + + return events_id_cmp(a, b); +} + +struct event_format **pevent_list_events(struct pevent *pevent, enum event_sort_type sort_type) +{ + struct event_format **events; + int (*sort)(const void *a, const void *b); + + events = pevent->sort_events; + + if (events && pevent->last_type == sort_type) + return events; + + if (!events) { + events = malloc(sizeof(*events) * (pevent->nr_events + 1)); + if (!events) + return NULL; + + memcpy(events, pevent->events, sizeof(*events) * pevent->nr_events); + events[pevent->nr_events] = NULL; + + pevent->sort_events = events; + + /* the internal events are sorted by id */ + if (sort_type == EVENT_SORT_ID) { + pevent->last_type = sort_type; + return events; + } + } + + switch (sort_type) { + case EVENT_SORT_ID: + sort = events_id_cmp; + break; + case EVENT_SORT_NAME: + sort = events_name_cmp; + break; + case EVENT_SORT_SYSTEM: + sort = events_system_cmp; + break; + default: + return events; + } + + qsort(events, pevent->nr_events, sizeof(*events), sort); + pevent->last_type = sort_type; + + return events; +} + +static struct format_field ** +get_event_fields(const char *type, const char *name, + int count, struct format_field *list) +{ + struct format_field **fields; + struct format_field *field; + int i = 0; + + fields = malloc(sizeof(*fields) * (count + 1)); + if (!fields) + return NULL; + + for (field = list; field; field = field->next) { + fields[i++] = field; + if (i == count + 1) { + do_warning("event %s has more %s fields than specified", + name, type); + i--; + break; + } + } + + if (i != count) + do_warning("event %s has less %s fields than specified", + name, type); + + fields[i] = NULL; + + return fields; +} + +/** + * pevent_event_common_fields - return a list of common fields for an event + * @event: the event to return the common fields of. + * + * Returns an allocated array of fields. The last item in the array is NULL. + * The array must be freed with free(). + */ +struct format_field **pevent_event_common_fields(struct event_format *event) +{ + return get_event_fields("common", event->name, + event->format.nr_common, + event->format.common_fields); +} + +/** + * pevent_event_fields - return a list of event specific fields for an event + * @event: the event to return the fields of. + * + * Returns an allocated array of fields. The last item in the array is NULL. + * The array must be freed with free(). + */ +struct format_field **pevent_event_fields(struct event_format *event) +{ + return get_event_fields("event", event->name, + event->format.nr_fields, + event->format.fields); +} + +static void print_fields(struct trace_seq *s, struct print_flag_sym *field) +{ + trace_seq_printf(s, "{ %s, %s }", field->value, field->str); + if (field->next) { + trace_seq_puts(s, ", "); + print_fields(s, field->next); + } +} + +/* for debugging */ +static void print_args(struct print_arg *args) +{ + int print_paren = 1; + struct trace_seq s; + + switch (args->type) { + case PRINT_NULL: + printf("null"); + break; + case PRINT_ATOM: + printf("%s", args->atom.atom); + break; + case PRINT_FIELD: + printf("REC->%s", args->field.name); + break; + case PRINT_FLAGS: + printf("__print_flags("); + print_args(args->flags.field); + printf(", %s, ", args->flags.delim); + trace_seq_init(&s); + print_fields(&s, args->flags.flags); + trace_seq_do_printf(&s); + trace_seq_destroy(&s); + printf(")"); + break; + case PRINT_SYMBOL: + printf("__print_symbolic("); + print_args(args->symbol.field); + printf(", "); + trace_seq_init(&s); + print_fields(&s, args->symbol.symbols); + trace_seq_do_printf(&s); + trace_seq_destroy(&s); + printf(")"); + break; + case PRINT_HEX: + printf("__print_hex("); + print_args(args->hex.field); + printf(", "); + print_args(args->hex.size); + printf(")"); + break; + case PRINT_STRING: + case PRINT_BSTRING: + printf("__get_str(%s)", args->string.string); + break; + case PRINT_TYPE: + printf("(%s)", args->typecast.type); + print_args(args->typecast.item); + break; + case PRINT_OP: + if (strcmp(args->op.op, ":") == 0) + print_paren = 0; + if (print_paren) + printf("("); + print_args(args->op.left); + printf(" %s ", args->op.op); + print_args(args->op.right); + if (print_paren) + printf(")"); + break; + default: + /* we should warn... */ + return; + } + if (args->next) { + printf("\n"); + print_args(args->next); + } +} + +static void parse_header_field(const char *field, + int *offset, int *size, int mandatory) +{ + unsigned long long save_input_buf_ptr; + unsigned long long save_input_buf_siz; + char *token; + int type; + + save_input_buf_ptr = input_buf_ptr; + save_input_buf_siz = input_buf_siz; + + if (read_expected(EVENT_ITEM, "field") < 0) + return; + if (read_expected(EVENT_OP, ":") < 0) + return; + + /* type */ + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto fail; + free_token(token); + + /* + * If this is not a mandatory field, then test it first. + */ + if (mandatory) { + if (read_expected(EVENT_ITEM, field) < 0) + return; + } else { + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto fail; + if (strcmp(token, field) != 0) + goto discard; + free_token(token); + } + + if (read_expected(EVENT_OP, ";") < 0) + return; + if (read_expected(EVENT_ITEM, "offset") < 0) + return; + if (read_expected(EVENT_OP, ":") < 0) + return; + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto fail; + *offset = atoi(token); + free_token(token); + if (read_expected(EVENT_OP, ";") < 0) + return; + if (read_expected(EVENT_ITEM, "size") < 0) + return; + if (read_expected(EVENT_OP, ":") < 0) + return; + if (read_expect_type(EVENT_ITEM, &token) < 0) + goto fail; + *size = atoi(token); + free_token(token); + if (read_expected(EVENT_OP, ";") < 0) + return; + type = read_token(&token); + if (type != EVENT_NEWLINE) { + /* newer versions of the kernel have a "signed" type */ + if (type != EVENT_ITEM) + goto fail; + + if (strcmp(token, "signed") != 0) + goto fail; + + free_token(token); + + if (read_expected(EVENT_OP, ":") < 0) + return; + + if (read_expect_type(EVENT_ITEM, &token)) + goto fail; + + free_token(token); + if (read_expected(EVENT_OP, ";") < 0) + return; + + if (read_expect_type(EVENT_NEWLINE, &token)) + goto fail; + } + fail: + free_token(token); + return; + + discard: + input_buf_ptr = save_input_buf_ptr; + input_buf_siz = save_input_buf_siz; + *offset = 0; + *size = 0; + free_token(token); +} + +/** + * pevent_parse_header_page - parse the data stored in the header page + * @pevent: the handle to the pevent + * @buf: the buffer storing the header page format string + * @size: the size of @buf + * @long_size: the long size to use if there is no header + * + * This parses the header page format for information on the + * ring buffer used. The @buf should be copied from + * + * /sys/kernel/debug/tracing/events/header_page + */ +int pevent_parse_header_page(struct pevent *pevent, char *buf, unsigned long size, + int long_size) +{ + int ignore; + + if (!size) { + /* + * Old kernels did not have header page info. + * Sorry but we just use what we find here in user space. + */ + pevent->header_page_ts_size = sizeof(long long); + pevent->header_page_size_size = long_size; + pevent->header_page_data_offset = sizeof(long long) + long_size; + pevent->old_format = 1; + return -1; + } + init_input_buf(buf, size); + + parse_header_field("timestamp", &pevent->header_page_ts_offset, + &pevent->header_page_ts_size, 1); + parse_header_field("commit", &pevent->header_page_size_offset, + &pevent->header_page_size_size, 1); + parse_header_field("overwrite", &pevent->header_page_overwrite, + &ignore, 0); + parse_header_field("data", &pevent->header_page_data_offset, + &pevent->header_page_data_size, 1); + + return 0; +} + +static int event_matches(struct event_format *event, + int id, const char *sys_name, + const char *event_name) +{ + if (id >= 0 && id != event->id) + return 0; + + if (event_name && (strcmp(event_name, event->name) != 0)) + return 0; + + if (sys_name && (strcmp(sys_name, event->system) != 0)) + return 0; + + return 1; +} + +static void free_handler(struct event_handler *handle) +{ + free((void *)handle->sys_name); + free((void *)handle->event_name); + free(handle); +} + +static int find_event_handle(struct pevent *pevent, struct event_format *event) +{ + struct event_handler *handle, **next; + + for (next = &pevent->handlers; *next; + next = &(*next)->next) { + handle = *next; + if (event_matches(event, handle->id, + handle->sys_name, + handle->event_name)) + break; + } + + if (!(*next)) + return 0; + + pr_stat("overriding event (%d) %s:%s with new print handler", + event->id, event->system, event->name); + + event->handler = handle->func; + event->context = handle->context; + + *next = handle->next; + free_handler(handle); + + return 1; +} + +/** + * __pevent_parse_format - parse the event format + * @buf: the buffer storing the event format string + * @size: the size of @buf + * @sys: the system the event belongs to + * + * This parses the event format and creates an event structure + * to quickly parse raw data for a given event. + * + * These files currently come from: + * + * /sys/kernel/debug/tracing/events/.../.../format + */ +enum pevent_errno __pevent_parse_format(struct event_format **eventp, + struct pevent *pevent, const char *buf, + unsigned long size, const char *sys) +{ + struct event_format *event; + int ret; + + init_input_buf(buf, size); + + *eventp = event = alloc_event(); + if (!event) + return PEVENT_ERRNO__MEM_ALLOC_FAILED; + + event->name = event_read_name(); + if (!event->name) { + /* Bad event? */ + ret = PEVENT_ERRNO__MEM_ALLOC_FAILED; + goto event_alloc_failed; + } + + if (strcmp(sys, "ftrace") == 0) { + event->flags |= EVENT_FL_ISFTRACE; + + if (strcmp(event->name, "bprint") == 0) + event->flags |= EVENT_FL_ISBPRINT; + } + + event->id = event_read_id(); + if (event->id < 0) { + ret = PEVENT_ERRNO__READ_ID_FAILED; + /* + * This isn't an allocation error actually. + * But as the ID is critical, just bail out. + */ + goto event_alloc_failed; + } + + event->system = strdup(sys); + if (!event->system) { + ret = PEVENT_ERRNO__MEM_ALLOC_FAILED; + goto event_alloc_failed; + } + + /* Add pevent to event so that it can be referenced */ + event->pevent = pevent; + + ret = event_read_format(event); + if (ret < 0) { + ret = PEVENT_ERRNO__READ_FORMAT_FAILED; + goto event_parse_failed; + } + + /* + * If the event has an override, don't print warnings if the event + * print format fails to parse. + */ + if (pevent && find_event_handle(pevent, event)) + show_warning = 0; + + ret = event_read_print(event); + show_warning = 1; + + if (ret < 0) { + ret = PEVENT_ERRNO__READ_PRINT_FAILED; + goto event_parse_failed; + } + + if (!ret && (event->flags & EVENT_FL_ISFTRACE)) { + struct format_field *field; + struct print_arg *arg, **list; + + /* old ftrace had no args */ + list = &event->print_fmt.args; + for (field = event->format.fields; field; field = field->next) { + arg = alloc_arg(); + if (!arg) { + event->flags |= EVENT_FL_FAILED; + return PEVENT_ERRNO__OLD_FTRACE_ARG_FAILED; + } + arg->type = PRINT_FIELD; + arg->field.name = strdup(field->name); + if (!arg->field.name) { + event->flags |= EVENT_FL_FAILED; + free_arg(arg); + return PEVENT_ERRNO__OLD_FTRACE_ARG_FAILED; + } + arg->field.field = field; + *list = arg; + list = &arg->next; + } + return 0; + } + + return 0; + + event_parse_failed: + event->flags |= EVENT_FL_FAILED; + return ret; + + event_alloc_failed: + free(event->system); + free(event->name); + free(event); + *eventp = NULL; + return ret; +} + +/** + * pevent_parse_format - parse the event format + * @buf: the buffer storing the event format string + * @size: the size of @buf + * @sys: the system the event belongs to + * + * This parses the event format and creates an event structure + * to quickly parse raw data for a given event. + * + * These files currently come from: + * + * /sys/kernel/debug/tracing/events/.../.../format + */ +enum pevent_errno pevent_parse_format(struct event_format **eventp, const char *buf, + unsigned long size, const char *sys) +{ + return __pevent_parse_format(eventp, NULL, buf, size, sys); +} + +/** + * pevent_parse_event - parse the event format + * @pevent: the handle to the pevent + * @buf: the buffer storing the event format string + * @size: the size of @buf + * @sys: the system the event belongs to + * + * This parses the event format and creates an event structure + * to quickly parse raw data for a given event. + * + * These files currently come from: + * + * /sys/kernel/debug/tracing/events/.../.../format + */ +enum pevent_errno pevent_parse_event(struct pevent *pevent, const char *buf, + unsigned long size, const char *sys) +{ + struct event_format *event = NULL; + int ret = __pevent_parse_format(&event, pevent, buf, size, sys); + + if (event == NULL) + return ret; + + if (add_event(pevent, event)) { + ret = PEVENT_ERRNO__MEM_ALLOC_FAILED; + goto event_add_failed; + } + +#define PRINT_ARGS 0 + if (PRINT_ARGS && event->print_fmt.args) + print_args(event->print_fmt.args); + + return 0; + +event_add_failed: + pevent_free_format(event); + return ret; +} + +#undef _PE +#define _PE(code, str) str +static const char * const pevent_error_str[] = { + PEVENT_ERRORS +}; +#undef _PE + +int pevent_strerror(struct pevent *pevent __maybe_unused, + enum pevent_errno errnum, char *buf, size_t buflen) +{ + int idx; + const char *msg; + + if (errnum >= 0) { + msg = strerror_r(errnum, buf, buflen); + if (msg != buf) { + size_t len = strlen(msg); + memcpy(buf, msg, min(buflen - 1, len)); + *(buf + min(buflen - 1, len)) = '\0'; + } + return 0; + } + + if (errnum <= __PEVENT_ERRNO__START || + errnum >= __PEVENT_ERRNO__END) + return -1; + + idx = errnum - __PEVENT_ERRNO__START - 1; + msg = pevent_error_str[idx]; + + switch (errnum) { + case PEVENT_ERRNO__MEM_ALLOC_FAILED: + case PEVENT_ERRNO__PARSE_EVENT_FAILED: + case PEVENT_ERRNO__READ_ID_FAILED: + case PEVENT_ERRNO__READ_FORMAT_FAILED: + case PEVENT_ERRNO__READ_PRINT_FAILED: + case PEVENT_ERRNO__OLD_FTRACE_ARG_FAILED: + case PEVENT_ERRNO__INVALID_ARG_TYPE: + snprintf(buf, buflen, "%s", msg); + break; + + default: + /* cannot reach here */ + break; + } + + return 0; +} + +int get_field_val(struct trace_seq *s, struct format_field *field, + const char *name, struct pevent_record *record, + unsigned long long *val, int err) +{ + if (!field) { + if (err) + trace_seq_printf(s, "", name); + return -1; + } + + if (pevent_read_number_field(field, record->data, val)) { + if (err) + trace_seq_printf(s, " %s=INVALID", name); + return -1; + } + + return 0; +} + +/** + * pevent_get_field_raw - return the raw pointer into the data field + * @s: The seq to print to on error + * @event: the event that the field is for + * @name: The name of the field + * @record: The record with the field name. + * @len: place to store the field length. + * @err: print default error if failed. + * + * Returns a pointer into record->data of the field and places + * the length of the field in @len. + * + * On failure, it returns NULL. + */ +void *pevent_get_field_raw(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + int *len, int err) +{ + struct format_field *field; + void *data = record->data; + unsigned offset; + int dummy; + + if (!event) + return NULL; + + field = pevent_find_field(event, name); + + if (!field) { + if (err) + trace_seq_printf(s, "", name); + return NULL; + } + + /* Allow @len to be NULL */ + if (!len) + len = &dummy; + + offset = field->offset; + if (field->flags & FIELD_IS_DYNAMIC) { + offset = pevent_read_number(event->pevent, + data + offset, field->size); + *len = offset >> 16; + offset &= 0xffff; + } else + *len = field->size; + + return data + offset; +} + +/** + * pevent_get_field_val - find a field and return its value + * @s: The seq to print to on error + * @event: the event that the field is for + * @name: The name of the field + * @record: The record with the field name. + * @val: place to store the value of the field. + * @err: print default error if failed. + * + * Returns 0 on success -1 on field not found. + */ +int pevent_get_field_val(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + unsigned long long *val, int err) +{ + struct format_field *field; + + if (!event) + return -1; + + field = pevent_find_field(event, name); + + return get_field_val(s, field, name, record, val, err); +} + +/** + * pevent_get_common_field_val - find a common field and return its value + * @s: The seq to print to on error + * @event: the event that the field is for + * @name: The name of the field + * @record: The record with the field name. + * @val: place to store the value of the field. + * @err: print default error if failed. + * + * Returns 0 on success -1 on field not found. + */ +int pevent_get_common_field_val(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + unsigned long long *val, int err) +{ + struct format_field *field; + + if (!event) + return -1; + + field = pevent_find_common_field(event, name); + + return get_field_val(s, field, name, record, val, err); +} + +/** + * pevent_get_any_field_val - find a any field and return its value + * @s: The seq to print to on error + * @event: the event that the field is for + * @name: The name of the field + * @record: The record with the field name. + * @val: place to store the value of the field. + * @err: print default error if failed. + * + * Returns 0 on success -1 on field not found. + */ +int pevent_get_any_field_val(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + unsigned long long *val, int err) +{ + struct format_field *field; + + if (!event) + return -1; + + field = pevent_find_any_field(event, name); + + return get_field_val(s, field, name, record, val, err); +} + +/** + * pevent_print_num_field - print a field and a format + * @s: The seq to print to + * @fmt: The printf format to print the field with. + * @event: the event that the field is for + * @name: The name of the field + * @record: The record with the field name. + * @err: print default error if failed. + * + * Returns: 0 on success, -1 field not found, or 1 if buffer is full. + */ +int pevent_print_num_field(struct trace_seq *s, const char *fmt, + struct event_format *event, const char *name, + struct pevent_record *record, int err) +{ + struct format_field *field = pevent_find_field(event, name); + unsigned long long val; + + if (!field) + goto failed; + + if (pevent_read_number_field(field, record->data, &val)) + goto failed; + + return trace_seq_printf(s, fmt, val); + + failed: + if (err) + trace_seq_printf(s, "CAN'T FIND FIELD \"%s\"", name); + return -1; +} + +static void free_func_handle(struct pevent_function_handler *func) +{ + struct pevent_func_params *params; + + free(func->name); + + while (func->params) { + params = func->params; + func->params = params->next; + free(params); + } + + free(func); +} + +/** + * pevent_register_print_function - register a helper function + * @pevent: the handle to the pevent + * @func: the function to process the helper function + * @ret_type: the return type of the helper function + * @name: the name of the helper function + * @parameters: A list of enum pevent_func_arg_type + * + * Some events may have helper functions in the print format arguments. + * This allows a plugin to dynamically create a way to process one + * of these functions. + * + * The @parameters is a variable list of pevent_func_arg_type enums that + * must end with PEVENT_FUNC_ARG_VOID. + */ +int pevent_register_print_function(struct pevent *pevent, + pevent_func_handler func, + enum pevent_func_arg_type ret_type, + char *name, ...) +{ + struct pevent_function_handler *func_handle; + struct pevent_func_params **next_param; + struct pevent_func_params *param; + enum pevent_func_arg_type type; + va_list ap; + int ret; + + func_handle = find_func_handler(pevent, name); + if (func_handle) { + /* + * This is most like caused by the users own + * plugins updating the function. This overrides the + * system defaults. + */ + pr_stat("override of function helper '%s'", name); + remove_func_handler(pevent, name); + } + + func_handle = calloc(1, sizeof(*func_handle)); + if (!func_handle) { + do_warning("Failed to allocate function handler"); + return PEVENT_ERRNO__MEM_ALLOC_FAILED; + } + + func_handle->ret_type = ret_type; + func_handle->name = strdup(name); + func_handle->func = func; + if (!func_handle->name) { + do_warning("Failed to allocate function name"); + free(func_handle); + return PEVENT_ERRNO__MEM_ALLOC_FAILED; + } + + next_param = &(func_handle->params); + va_start(ap, name); + for (;;) { + type = va_arg(ap, enum pevent_func_arg_type); + if (type == PEVENT_FUNC_ARG_VOID) + break; + + if (type >= PEVENT_FUNC_ARG_MAX_TYPES) { + do_warning("Invalid argument type %d", type); + ret = PEVENT_ERRNO__INVALID_ARG_TYPE; + goto out_free; + } + + param = malloc(sizeof(*param)); + if (!param) { + do_warning("Failed to allocate function param"); + ret = PEVENT_ERRNO__MEM_ALLOC_FAILED; + goto out_free; + } + param->type = type; + param->next = NULL; + + *next_param = param; + next_param = &(param->next); + + func_handle->nr_args++; + } + va_end(ap); + + func_handle->next = pevent->func_handlers; + pevent->func_handlers = func_handle; + + return 0; + out_free: + va_end(ap); + free_func_handle(func_handle); + return ret; +} + +/** + * pevent_register_event_handler - register a way to parse an event + * @pevent: the handle to the pevent + * @id: the id of the event to register + * @sys_name: the system name the event belongs to + * @event_name: the name of the event + * @func: the function to call to parse the event information + * @context: the data to be passed to @func + * + * This function allows a developer to override the parsing of + * a given event. If for some reason the default print format + * is not sufficient, this function will register a function + * for an event to be used to parse the data instead. + * + * If @id is >= 0, then it is used to find the event. + * else @sys_name and @event_name are used. + */ +int pevent_register_event_handler(struct pevent *pevent, + int id, char *sys_name, char *event_name, + pevent_event_handler_func func, + void *context) +{ + struct event_format *event; + struct event_handler *handle; + + if (id >= 0) { + /* search by id */ + event = pevent_find_event(pevent, id); + if (!event) + goto not_found; + if (event_name && (strcmp(event_name, event->name) != 0)) + goto not_found; + if (sys_name && (strcmp(sys_name, event->system) != 0)) + goto not_found; + } else { + event = pevent_find_event_by_name(pevent, sys_name, event_name); + if (!event) + goto not_found; + } + + pr_stat("overriding event (%d) %s:%s with new print handler", + event->id, event->system, event->name); + + event->handler = func; + event->context = context; + return 0; + + not_found: + /* Save for later use. */ + handle = calloc(1, sizeof(*handle)); + if (!handle) { + do_warning("Failed to allocate event handler"); + return PEVENT_ERRNO__MEM_ALLOC_FAILED; + } + + handle->id = id; + if (event_name) + handle->event_name = strdup(event_name); + if (sys_name) + handle->sys_name = strdup(sys_name); + + if ((event_name && !handle->event_name) || + (sys_name && !handle->sys_name)) { + do_warning("Failed to allocate event/sys name"); + free((void *)handle->event_name); + free((void *)handle->sys_name); + free(handle); + return PEVENT_ERRNO__MEM_ALLOC_FAILED; + } + + handle->func = func; + handle->next = pevent->handlers; + pevent->handlers = handle; + handle->context = context; + + return -1; +} + +/** + * pevent_alloc - create a pevent handle + */ +struct pevent *pevent_alloc(void) +{ + struct pevent *pevent = calloc(1, sizeof(*pevent)); + + if (pevent) + pevent->ref_count = 1; + + return pevent; +} + +void pevent_ref(struct pevent *pevent) +{ + pevent->ref_count++; +} + +static void free_format_fields(struct format_field *field) +{ + struct format_field *next; + + while (field) { + next = field->next; + free(field->type); + free(field->name); + free(field); + field = next; + } +} + +static void free_formats(struct format *format) +{ + free_format_fields(format->common_fields); + free_format_fields(format->fields); +} + +void pevent_free_format(struct event_format *event) +{ + free(event->name); + free(event->system); + + free_formats(&event->format); + + free(event->print_fmt.format); + free_args(event->print_fmt.args); + + free(event); +} + +/** + * pevent_free - free a pevent handle + * @pevent: the pevent handle to free + */ +void pevent_free(struct pevent *pevent) +{ + struct cmdline_list *cmdlist, *cmdnext; + struct func_list *funclist, *funcnext; + struct printk_list *printklist, *printknext; + struct pevent_function_handler *func_handler; + struct event_handler *handle; + int i; + + if (!pevent) + return; + + cmdlist = pevent->cmdlist; + funclist = pevent->funclist; + printklist = pevent->printklist; + + pevent->ref_count--; + if (pevent->ref_count) + return; + + if (pevent->cmdlines) { + for (i = 0; i < pevent->cmdline_count; i++) + free(pevent->cmdlines[i].comm); + free(pevent->cmdlines); + } + + while (cmdlist) { + cmdnext = cmdlist->next; + free(cmdlist->comm); + free(cmdlist); + cmdlist = cmdnext; + } + + if (pevent->func_map) { + for (i = 0; i < (int)pevent->func_count; i++) { + free(pevent->func_map[i].func); + free(pevent->func_map[i].mod); + } + free(pevent->func_map); + } + + while (funclist) { + funcnext = funclist->next; + free(funclist->func); + free(funclist->mod); + free(funclist); + funclist = funcnext; + } + + while (pevent->func_handlers) { + func_handler = pevent->func_handlers; + pevent->func_handlers = func_handler->next; + free_func_handle(func_handler); + } + + if (pevent->printk_map) { + for (i = 0; i < (int)pevent->printk_count; i++) + free(pevent->printk_map[i].printk); + free(pevent->printk_map); + } + + while (printklist) { + printknext = printklist->next; + free(printklist->printk); + free(printklist); + printklist = printknext; + } + + for (i = 0; i < pevent->nr_events; i++) + pevent_free_format(pevent->events[i]); + + while (pevent->handlers) { + handle = pevent->handlers; + pevent->handlers = handle->next; + free_handler(handle); + } + + free(pevent->events); + free(pevent->sort_events); + + free(pevent); +} + +void pevent_unref(struct pevent *pevent) +{ + pevent_free(pevent); +} diff --git a/tools/lib/traceevent/event-parse.h b/tools/lib/traceevent/event-parse.h new file mode 100644 index 000000000..7be7e8953 --- /dev/null +++ b/tools/lib/traceevent/event-parse.h @@ -0,0 +1,844 @@ +/* + * Copyright (C) 2009, 2010 Red Hat Inc, Steven Rostedt + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License (not later!) + * + * 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this program; if not, see + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + */ +#ifndef _PARSE_EVENTS_H +#define _PARSE_EVENTS_H + +#include +#include + +#ifndef __maybe_unused +#define __maybe_unused __attribute__((unused)) +#endif + +/* ----------------------- trace_seq ----------------------- */ + + +#ifndef TRACE_SEQ_BUF_SIZE +#define TRACE_SEQ_BUF_SIZE 4096 +#endif + +#ifndef DEBUG_RECORD +#define DEBUG_RECORD 0 +#endif + +struct pevent_record { + unsigned long long ts; + unsigned long long offset; + long long missed_events; /* buffer dropped events before */ + int record_size; /* size of binary record */ + int size; /* size of data */ + void *data; + int cpu; + int ref_count; + int locked; /* Do not free, even if ref_count is zero */ + void *priv; +#if DEBUG_RECORD + struct pevent_record *prev; + struct pevent_record *next; + long alloc_addr; +#endif +}; + +/* + * Trace sequences are used to allow a function to call several other functions + * to create a string of data to use (up to a max of PAGE_SIZE). + */ + +struct trace_seq { + char *buffer; + unsigned int buffer_size; + unsigned int len; + unsigned int readpos; +}; + +void trace_seq_init(struct trace_seq *s); +void trace_seq_destroy(struct trace_seq *s); + +extern int trace_seq_printf(struct trace_seq *s, const char *fmt, ...) + __attribute__ ((format (printf, 2, 3))); +extern int trace_seq_vprintf(struct trace_seq *s, const char *fmt, va_list args) + __attribute__ ((format (printf, 2, 0))); + +extern int trace_seq_puts(struct trace_seq *s, const char *str); +extern int trace_seq_putc(struct trace_seq *s, unsigned char c); + +extern void trace_seq_terminate(struct trace_seq *s); + +extern int trace_seq_do_printf(struct trace_seq *s); + + +/* ----------------------- pevent ----------------------- */ + +struct pevent; +struct event_format; + +typedef int (*pevent_event_handler_func)(struct trace_seq *s, + struct pevent_record *record, + struct event_format *event, + void *context); + +typedef int (*pevent_plugin_load_func)(struct pevent *pevent); +typedef int (*pevent_plugin_unload_func)(void); + +struct plugin_option { + struct plugin_option *next; + void *handle; + char *file; + char *name; + char *plugin_alias; + char *description; + char *value; + void *priv; + int set; +}; + +/* + * Plugin hooks that can be called: + * + * PEVENT_PLUGIN_LOADER: (required) + * The function name to initialized the plugin. + * + * int PEVENT_PLUGIN_LOADER(struct pevent *pevent) + * + * PEVENT_PLUGIN_UNLOADER: (optional) + * The function called just before unloading + * + * int PEVENT_PLUGIN_UNLOADER(void) + * + * PEVENT_PLUGIN_OPTIONS: (optional) + * Plugin options that can be set before loading + * + * struct plugin_option PEVENT_PLUGIN_OPTIONS[] = { + * { + * .name = "option-name", + * .plugin_alias = "overide-file-name", (optional) + * .description = "description of option to show users", + * }, + * { + * .name = NULL, + * }, + * }; + * + * Array must end with .name = NULL; + * + * + * .plugin_alias is used to give a shorter name to access + * the vairable. Useful if a plugin handles more than one event. + * + * PEVENT_PLUGIN_ALIAS: (optional) + * The name to use for finding options (uses filename if not defined) + */ +#define PEVENT_PLUGIN_LOADER pevent_plugin_loader +#define PEVENT_PLUGIN_UNLOADER pevent_plugin_unloader +#define PEVENT_PLUGIN_OPTIONS pevent_plugin_options +#define PEVENT_PLUGIN_ALIAS pevent_plugin_alias +#define _MAKE_STR(x) #x +#define MAKE_STR(x) _MAKE_STR(x) +#define PEVENT_PLUGIN_LOADER_NAME MAKE_STR(PEVENT_PLUGIN_LOADER) +#define PEVENT_PLUGIN_UNLOADER_NAME MAKE_STR(PEVENT_PLUGIN_UNLOADER) +#define PEVENT_PLUGIN_OPTIONS_NAME MAKE_STR(PEVENT_PLUGIN_OPTIONS) +#define PEVENT_PLUGIN_ALIAS_NAME MAKE_STR(PEVENT_PLUGIN_ALIAS) + +#define NSECS_PER_SEC 1000000000ULL +#define NSECS_PER_USEC 1000ULL + +enum format_flags { + FIELD_IS_ARRAY = 1, + FIELD_IS_POINTER = 2, + FIELD_IS_SIGNED = 4, + FIELD_IS_STRING = 8, + FIELD_IS_DYNAMIC = 16, + FIELD_IS_LONG = 32, + FIELD_IS_FLAG = 64, + FIELD_IS_SYMBOLIC = 128, +}; + +struct format_field { + struct format_field *next; + struct event_format *event; + char *type; + char *name; + int offset; + int size; + unsigned int arraylen; + unsigned int elementsize; + unsigned long flags; +}; + +struct format { + int nr_common; + int nr_fields; + struct format_field *common_fields; + struct format_field *fields; +}; + +struct print_arg_atom { + char *atom; +}; + +struct print_arg_string { + char *string; + int offset; +}; + +struct print_arg_field { + char *name; + struct format_field *field; +}; + +struct print_flag_sym { + struct print_flag_sym *next; + char *value; + char *str; +}; + +struct print_arg_typecast { + char *type; + struct print_arg *item; +}; + +struct print_arg_flags { + struct print_arg *field; + char *delim; + struct print_flag_sym *flags; +}; + +struct print_arg_symbol { + struct print_arg *field; + struct print_flag_sym *symbols; +}; + +struct print_arg_hex { + struct print_arg *field; + struct print_arg *size; +}; + +struct print_arg_dynarray { + struct format_field *field; + struct print_arg *index; +}; + +struct print_arg; + +struct print_arg_op { + char *op; + int prio; + struct print_arg *left; + struct print_arg *right; +}; + +struct pevent_function_handler; + +struct print_arg_func { + struct pevent_function_handler *func; + struct print_arg *args; +}; + +enum print_arg_type { + PRINT_NULL, + PRINT_ATOM, + PRINT_FIELD, + PRINT_FLAGS, + PRINT_SYMBOL, + PRINT_HEX, + PRINT_TYPE, + PRINT_STRING, + PRINT_BSTRING, + PRINT_DYNAMIC_ARRAY, + PRINT_OP, + PRINT_FUNC, +}; + +struct print_arg { + struct print_arg *next; + enum print_arg_type type; + union { + struct print_arg_atom atom; + struct print_arg_field field; + struct print_arg_typecast typecast; + struct print_arg_flags flags; + struct print_arg_symbol symbol; + struct print_arg_hex hex; + struct print_arg_func func; + struct print_arg_string string; + struct print_arg_op op; + struct print_arg_dynarray dynarray; + }; +}; + +struct print_fmt { + char *format; + struct print_arg *args; +}; + +struct event_format { + struct pevent *pevent; + char *name; + int id; + int flags; + struct format format; + struct print_fmt print_fmt; + char *system; + pevent_event_handler_func handler; + void *context; +}; + +enum { + EVENT_FL_ISFTRACE = 0x01, + EVENT_FL_ISPRINT = 0x02, + EVENT_FL_ISBPRINT = 0x04, + EVENT_FL_ISFUNCENT = 0x10, + EVENT_FL_ISFUNCRET = 0x20, + + EVENT_FL_FAILED = 0x80000000 +}; + +enum event_sort_type { + EVENT_SORT_ID, + EVENT_SORT_NAME, + EVENT_SORT_SYSTEM, +}; + +enum event_type { + EVENT_ERROR, + EVENT_NONE, + EVENT_SPACE, + EVENT_NEWLINE, + EVENT_OP, + EVENT_DELIM, + EVENT_ITEM, + EVENT_DQUOTE, + EVENT_SQUOTE, +}; + +typedef unsigned long long (*pevent_func_handler)(struct trace_seq *s, + unsigned long long *args); + +enum pevent_func_arg_type { + PEVENT_FUNC_ARG_VOID, + PEVENT_FUNC_ARG_INT, + PEVENT_FUNC_ARG_LONG, + PEVENT_FUNC_ARG_STRING, + PEVENT_FUNC_ARG_PTR, + PEVENT_FUNC_ARG_MAX_TYPES +}; + +enum pevent_flag { + PEVENT_NSEC_OUTPUT = 1, /* output in NSECS */ +}; + +#define PEVENT_ERRORS \ + _PE(MEM_ALLOC_FAILED, "failed to allocate memory"), \ + _PE(PARSE_EVENT_FAILED, "failed to parse event"), \ + _PE(READ_ID_FAILED, "failed to read event id"), \ + _PE(READ_FORMAT_FAILED, "failed to read event format"), \ + _PE(READ_PRINT_FAILED, "failed to read event print fmt"), \ + _PE(OLD_FTRACE_ARG_FAILED,"failed to allocate field name for ftrace"),\ + _PE(INVALID_ARG_TYPE, "invalid argument type") + +#undef _PE +#define _PE(__code, __str) PEVENT_ERRNO__ ## __code +enum pevent_errno { + PEVENT_ERRNO__SUCCESS = 0, + + /* + * Choose an arbitrary negative big number not to clash with standard + * errno since SUS requires the errno has distinct positive values. + * See 'Issue 6' in the link below. + * + * http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/errno.h.html + */ + __PEVENT_ERRNO__START = -100000, + + PEVENT_ERRORS, + + __PEVENT_ERRNO__END, +}; +#undef _PE + +struct cmdline; +struct cmdline_list; +struct func_map; +struct func_list; +struct event_handler; + +struct pevent { + int ref_count; + + int header_page_ts_offset; + int header_page_ts_size; + int header_page_size_offset; + int header_page_size_size; + int header_page_data_offset; + int header_page_data_size; + int header_page_overwrite; + + int file_bigendian; + int host_bigendian; + + int latency_format; + + int old_format; + + int cpus; + int long_size; + + struct cmdline *cmdlines; + struct cmdline_list *cmdlist; + int cmdline_count; + + struct func_map *func_map; + struct func_list *funclist; + unsigned int func_count; + + struct printk_map *printk_map; + struct printk_list *printklist; + unsigned int printk_count; + + + struct event_format **events; + int nr_events; + struct event_format **sort_events; + enum event_sort_type last_type; + + int type_offset; + int type_size; + + int pid_offset; + int pid_size; + + int pc_offset; + int pc_size; + + int flags_offset; + int flags_size; + + int ld_offset; + int ld_size; + + int print_raw; + + int test_filters; + + int flags; + + struct format_field *bprint_ip_field; + struct format_field *bprint_fmt_field; + struct format_field *bprint_buf_field; + + struct event_handler *handlers; + struct pevent_function_handler *func_handlers; + + /* cache */ + struct event_format *last_event; +}; + +static inline void pevent_set_flag(struct pevent *pevent, int flag) +{ + pevent->flags |= flag; +} + +static inline unsigned short +__data2host2(struct pevent *pevent, unsigned short data) +{ + unsigned short swap; + + if (pevent->host_bigendian == pevent->file_bigendian) + return data; + + swap = ((data & 0xffULL) << 8) | + ((data & (0xffULL << 8)) >> 8); + + return swap; +} + +static inline unsigned int +__data2host4(struct pevent *pevent, unsigned int data) +{ + unsigned int swap; + + if (pevent->host_bigendian == pevent->file_bigendian) + return data; + + swap = ((data & 0xffULL) << 24) | + ((data & (0xffULL << 8)) << 8) | + ((data & (0xffULL << 16)) >> 8) | + ((data & (0xffULL << 24)) >> 24); + + return swap; +} + +static inline unsigned long long +__data2host8(struct pevent *pevent, unsigned long long data) +{ + unsigned long long swap; + + if (pevent->host_bigendian == pevent->file_bigendian) + return data; + + swap = ((data & 0xffULL) << 56) | + ((data & (0xffULL << 8)) << 40) | + ((data & (0xffULL << 16)) << 24) | + ((data & (0xffULL << 24)) << 8) | + ((data & (0xffULL << 32)) >> 8) | + ((data & (0xffULL << 40)) >> 24) | + ((data & (0xffULL << 48)) >> 40) | + ((data & (0xffULL << 56)) >> 56); + + return swap; +} + +#define data2host2(pevent, ptr) __data2host2(pevent, *(unsigned short *)(ptr)) +#define data2host4(pevent, ptr) __data2host4(pevent, *(unsigned int *)(ptr)) +#define data2host8(pevent, ptr) \ +({ \ + unsigned long long __val; \ + \ + memcpy(&__val, (ptr), sizeof(unsigned long long)); \ + __data2host8(pevent, __val); \ +}) + +/* taken from kernel/trace/trace.h */ +enum trace_flag_type { + TRACE_FLAG_IRQS_OFF = 0x01, + TRACE_FLAG_IRQS_NOSUPPORT = 0x02, + TRACE_FLAG_NEED_RESCHED = 0x04, + TRACE_FLAG_HARDIRQ = 0x08, + TRACE_FLAG_SOFTIRQ = 0x10, +}; + +int pevent_register_comm(struct pevent *pevent, const char *comm, int pid); +int pevent_register_function(struct pevent *pevent, char *name, + unsigned long long addr, char *mod); +int pevent_register_print_string(struct pevent *pevent, char *fmt, + unsigned long long addr); +int pevent_pid_is_registered(struct pevent *pevent, int pid); + +void pevent_print_event(struct pevent *pevent, struct trace_seq *s, + struct pevent_record *record); + +int pevent_parse_header_page(struct pevent *pevent, char *buf, unsigned long size, + int long_size); + +enum pevent_errno pevent_parse_event(struct pevent *pevent, const char *buf, + unsigned long size, const char *sys); +enum pevent_errno pevent_parse_format(struct event_format **eventp, const char *buf, + unsigned long size, const char *sys); +void pevent_free_format(struct event_format *event); + +void *pevent_get_field_raw(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + int *len, int err); + +int pevent_get_field_val(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + unsigned long long *val, int err); +int pevent_get_common_field_val(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + unsigned long long *val, int err); +int pevent_get_any_field_val(struct trace_seq *s, struct event_format *event, + const char *name, struct pevent_record *record, + unsigned long long *val, int err); + +int pevent_print_num_field(struct trace_seq *s, const char *fmt, + struct event_format *event, const char *name, + struct pevent_record *record, int err); + +int pevent_register_event_handler(struct pevent *pevent, int id, char *sys_name, char *event_name, + pevent_event_handler_func func, void *context); +int pevent_register_print_function(struct pevent *pevent, + pevent_func_handler func, + enum pevent_func_arg_type ret_type, + char *name, ...); + +struct format_field *pevent_find_common_field(struct event_format *event, const char *name); +struct format_field *pevent_find_field(struct event_format *event, const char *name); +struct format_field *pevent_find_any_field(struct event_format *event, const char *name); + +const char *pevent_find_function(struct pevent *pevent, unsigned long long addr); +unsigned long long +pevent_find_function_address(struct pevent *pevent, unsigned long long addr); +unsigned long long pevent_read_number(struct pevent *pevent, const void *ptr, int size); +int pevent_read_number_field(struct format_field *field, const void *data, + unsigned long long *value); + +struct event_format *pevent_find_event(struct pevent *pevent, int id); + +struct event_format * +pevent_find_event_by_name(struct pevent *pevent, const char *sys, const char *name); + +void pevent_data_lat_fmt(struct pevent *pevent, + struct trace_seq *s, struct pevent_record *record); +int pevent_data_type(struct pevent *pevent, struct pevent_record *rec); +struct event_format *pevent_data_event_from_type(struct pevent *pevent, int type); +int pevent_data_pid(struct pevent *pevent, struct pevent_record *rec); +const char *pevent_data_comm_from_pid(struct pevent *pevent, int pid); +void pevent_event_info(struct trace_seq *s, struct event_format *event, + struct pevent_record *record); +int pevent_strerror(struct pevent *pevent, enum pevent_errno errnum, + char *buf, size_t buflen); + +struct event_format **pevent_list_events(struct pevent *pevent, enum event_sort_type); +struct format_field **pevent_event_common_fields(struct event_format *event); +struct format_field **pevent_event_fields(struct event_format *event); + +static inline int pevent_get_cpus(struct pevent *pevent) +{ + return pevent->cpus; +} + +static inline void pevent_set_cpus(struct pevent *pevent, int cpus) +{ + pevent->cpus = cpus; +} + +static inline int pevent_get_long_size(struct pevent *pevent) +{ + return pevent->long_size; +} + +static inline void pevent_set_long_size(struct pevent *pevent, int long_size) +{ + pevent->long_size = long_size; +} + +static inline int pevent_is_file_bigendian(struct pevent *pevent) +{ + return pevent->file_bigendian; +} + +static inline void pevent_set_file_bigendian(struct pevent *pevent, int endian) +{ + pevent->file_bigendian = endian; +} + +static inline int pevent_is_host_bigendian(struct pevent *pevent) +{ + return pevent->host_bigendian; +} + +static inline void pevent_set_host_bigendian(struct pevent *pevent, int endian) +{ + pevent->host_bigendian = endian; +} + +static inline int pevent_is_latency_format(struct pevent *pevent) +{ + return pevent->latency_format; +} + +static inline void pevent_set_latency_format(struct pevent *pevent, int lat) +{ + pevent->latency_format = lat; +} + +struct pevent *pevent_alloc(void); +void pevent_free(struct pevent *pevent); +void pevent_ref(struct pevent *pevent); +void pevent_unref(struct pevent *pevent); + +/* access to the internal parser */ +void pevent_buffer_init(const char *buf, unsigned long long size); +enum event_type pevent_read_token(char **tok); +void pevent_free_token(char *token); +int pevent_peek_char(void); +const char *pevent_get_input_buf(void); +unsigned long long pevent_get_input_buf_ptr(void); + +/* for debugging */ +void pevent_print_funcs(struct pevent *pevent); +void pevent_print_printk(struct pevent *pevent); + +/* ----------------------- filtering ----------------------- */ + +enum filter_boolean_type { + FILTER_FALSE, + FILTER_TRUE, +}; + +enum filter_op_type { + FILTER_OP_AND = 1, + FILTER_OP_OR, + FILTER_OP_NOT, +}; + +enum filter_cmp_type { + FILTER_CMP_NONE, + FILTER_CMP_EQ, + FILTER_CMP_NE, + FILTER_CMP_GT, + FILTER_CMP_LT, + FILTER_CMP_GE, + FILTER_CMP_LE, + FILTER_CMP_MATCH, + FILTER_CMP_NOT_MATCH, + FILTER_CMP_REGEX, + FILTER_CMP_NOT_REGEX, +}; + +enum filter_exp_type { + FILTER_EXP_NONE, + FILTER_EXP_ADD, + FILTER_EXP_SUB, + FILTER_EXP_MUL, + FILTER_EXP_DIV, + FILTER_EXP_MOD, + FILTER_EXP_RSHIFT, + FILTER_EXP_LSHIFT, + FILTER_EXP_AND, + FILTER_EXP_OR, + FILTER_EXP_XOR, + FILTER_EXP_NOT, +}; + +enum filter_arg_type { + FILTER_ARG_NONE, + FILTER_ARG_BOOLEAN, + FILTER_ARG_VALUE, + FILTER_ARG_FIELD, + FILTER_ARG_EXP, + FILTER_ARG_OP, + FILTER_ARG_NUM, + FILTER_ARG_STR, +}; + +enum filter_value_type { + FILTER_NUMBER, + FILTER_STRING, + FILTER_CHAR +}; + +struct fliter_arg; + +struct filter_arg_boolean { + enum filter_boolean_type value; +}; + +struct filter_arg_field { + struct format_field *field; +}; + +struct filter_arg_value { + enum filter_value_type type; + union { + char *str; + unsigned long long val; + }; +}; + +struct filter_arg_op { + enum filter_op_type type; + struct filter_arg *left; + struct filter_arg *right; +}; + +struct filter_arg_exp { + enum filter_exp_type type; + struct filter_arg *left; + struct filter_arg *right; +}; + +struct filter_arg_num { + enum filter_cmp_type type; + struct filter_arg *left; + struct filter_arg *right; +}; + +struct filter_arg_str { + enum filter_cmp_type type; + struct format_field *field; + char *val; + char *buffer; + regex_t reg; +}; + +struct filter_arg { + enum filter_arg_type type; + union { + struct filter_arg_boolean boolean; + struct filter_arg_field field; + struct filter_arg_value value; + struct filter_arg_op op; + struct filter_arg_exp exp; + struct filter_arg_num num; + struct filter_arg_str str; + }; +}; + +struct filter_type { + int event_id; + struct event_format *event; + struct filter_arg *filter; +}; + +struct event_filter { + struct pevent *pevent; + int filters; + struct filter_type *event_filters; +}; + +struct event_filter *pevent_filter_alloc(struct pevent *pevent); + +#define FILTER_NONE -2 +#define FILTER_NOEXIST -1 +#define FILTER_MISS 0 +#define FILTER_MATCH 1 + +enum filter_trivial_type { + FILTER_TRIVIAL_FALSE, + FILTER_TRIVIAL_TRUE, + FILTER_TRIVIAL_BOTH, +}; + +int pevent_filter_add_filter_str(struct event_filter *filter, + const char *filter_str, + char **error_str); + + +int pevent_filter_match(struct event_filter *filter, + struct pevent_record *record); + +int pevent_event_filtered(struct event_filter *filter, + int event_id); + +void pevent_filter_reset(struct event_filter *filter); + +void pevent_filter_clear_trivial(struct event_filter *filter, + enum filter_trivial_type type); + +void pevent_filter_free(struct event_filter *filter); + +char *pevent_filter_make_string(struct event_filter *filter, int event_id); + +int pevent_filter_remove_event(struct event_filter *filter, + int event_id); + +int pevent_filter_event_has_trivial(struct event_filter *filter, + int event_id, + enum filter_trivial_type type); + +int pevent_filter_copy(struct event_filter *dest, struct event_filter *source); + +int pevent_update_trivial(struct event_filter *dest, struct event_filter *source, + enum filter_trivial_type type); + +int pevent_filter_compare(struct event_filter *filter1, struct event_filter *filter2); + +#endif /* _PARSE_EVENTS_H */ diff --git a/tools/lib/traceevent/event-utils.h b/tools/lib/traceevent/event-utils.h new file mode 100644 index 000000000..e76c9acb9 --- /dev/null +++ b/tools/lib/traceevent/event-utils.h @@ -0,0 +1,85 @@ +/* + * Copyright (C) 2010 Red Hat Inc, Steven Rostedt + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License (not later!) + * + * 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this program; if not, see + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + */ +#ifndef __UTIL_H +#define __UTIL_H + +#include + +/* Can be overridden */ +void die(const char *fmt, ...); +void *malloc_or_die(unsigned int size); +void warning(const char *fmt, ...); +void pr_stat(const char *fmt, ...); +void vpr_stat(const char *fmt, va_list ap); + +/* Always available */ +void __die(const char *fmt, ...); +void __warning(const char *fmt, ...); +void __pr_stat(const char *fmt, ...); + +void __vdie(const char *fmt, ...); +void __vwarning(const char *fmt, ...); +void __vpr_stat(const char *fmt, ...); + +#define min(x, y) ({ \ + typeof(x) _min1 = (x); \ + typeof(y) _min2 = (y); \ + (void) (&_min1 == &_min2); \ + _min1 < _min2 ? _min1 : _min2; }) + +static inline char *strim(char *string) +{ + char *ret; + + if (!string) + return NULL; + while (*string) { + if (!isspace(*string)) + break; + string++; + } + ret = string; + + string = ret + strlen(ret) - 1; + while (string > ret) { + if (!isspace(*string)) + break; + string--; + } + string[1] = 0; + + return ret; +} + +static inline int has_text(const char *text) +{ + if (!text) + return 0; + + while (*text) { + if (!isspace(*text)) + return 1; + text++; + } + + return 0; +} + +#endif diff --git a/tools/lib/traceevent/parse-filter.c b/tools/lib/traceevent/parse-filter.c new file mode 100644 index 000000000..2500e7558 --- /dev/null +++ b/tools/lib/traceevent/parse-filter.c @@ -0,0 +1,2303 @@ +/* + * Copyright (C) 2010 Red Hat Inc, Steven Rostedt + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License (not later!) + * + * 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this program; if not, see + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + */ +#include +#include +#include +#include +#include +#include + +#include "event-parse.h" +#include "event-utils.h" + +#define COMM "COMM" + +static struct format_field comm = { + .name = "COMM", +}; + +struct event_list { + struct event_list *next; + struct event_format *event; +}; + +#define MAX_ERR_STR_SIZE 256 + +static void show_error(char **error_str, const char *fmt, ...) +{ + unsigned long long index; + const char *input; + char *error; + va_list ap; + int len; + int i; + + if (!error_str) + return; + + input = pevent_get_input_buf(); + index = pevent_get_input_buf_ptr(); + len = input ? strlen(input) : 0; + + error = malloc_or_die(MAX_ERR_STR_SIZE + (len*2) + 3); + + if (len) { + strcpy(error, input); + error[len] = '\n'; + for (i = 1; i < len && i < index; i++) + error[len+i] = ' '; + error[len + i] = '^'; + error[len + i + 1] = '\n'; + len += i+2; + } + + va_start(ap, fmt); + vsnprintf(error + len, MAX_ERR_STR_SIZE, fmt, ap); + va_end(ap); + + *error_str = error; +} + +static void free_token(char *token) +{ + pevent_free_token(token); +} + +static enum event_type read_token(char **tok) +{ + enum event_type type; + char *token = NULL; + + do { + free_token(token); + type = pevent_read_token(&token); + } while (type == EVENT_NEWLINE || type == EVENT_SPACE); + + /* If token is = or ! check to see if the next char is ~ */ + if (token && + (strcmp(token, "=") == 0 || strcmp(token, "!") == 0) && + pevent_peek_char() == '~') { + /* append it */ + *tok = malloc_or_die(3); + sprintf(*tok, "%c%c", *token, '~'); + free_token(token); + /* Now remove the '~' from the buffer */ + pevent_read_token(&token); + free_token(token); + } else + *tok = token; + + return type; +} + +static int filter_cmp(const void *a, const void *b) +{ + const struct filter_type *ea = a; + const struct filter_type *eb = b; + + if (ea->event_id < eb->event_id) + return -1; + + if (ea->event_id > eb->event_id) + return 1; + + return 0; +} + +static struct filter_type * +find_filter_type(struct event_filter *filter, int id) +{ + struct filter_type *filter_type; + struct filter_type key; + + key.event_id = id; + + filter_type = bsearch(&key, filter->event_filters, + filter->filters, + sizeof(*filter->event_filters), + filter_cmp); + + return filter_type; +} + +static struct filter_type * +add_filter_type(struct event_filter *filter, int id) +{ + struct filter_type *filter_type; + int i; + + filter_type = find_filter_type(filter, id); + if (filter_type) + return filter_type; + + filter->event_filters = realloc(filter->event_filters, + sizeof(*filter->event_filters) * + (filter->filters + 1)); + if (!filter->event_filters) + die("Could not allocate filter"); + + for (i = 0; i < filter->filters; i++) { + if (filter->event_filters[i].event_id > id) + break; + } + + if (i < filter->filters) + memmove(&filter->event_filters[i+1], + &filter->event_filters[i], + sizeof(*filter->event_filters) * + (filter->filters - i)); + + filter_type = &filter->event_filters[i]; + filter_type->event_id = id; + filter_type->event = pevent_find_event(filter->pevent, id); + filter_type->filter = NULL; + + filter->filters++; + + return filter_type; +} + +/** + * pevent_filter_alloc - create a new event filter + * @pevent: The pevent that this filter is associated with + */ +struct event_filter *pevent_filter_alloc(struct pevent *pevent) +{ + struct event_filter *filter; + + filter = malloc_or_die(sizeof(*filter)); + memset(filter, 0, sizeof(*filter)); + filter->pevent = pevent; + pevent_ref(pevent); + + return filter; +} + +static struct filter_arg *allocate_arg(void) +{ + struct filter_arg *arg; + + arg = malloc_or_die(sizeof(*arg)); + memset(arg, 0, sizeof(*arg)); + + return arg; +} + +static void free_arg(struct filter_arg *arg) +{ + if (!arg) + return; + + switch (arg->type) { + case FILTER_ARG_NONE: + case FILTER_ARG_BOOLEAN: + break; + + case FILTER_ARG_NUM: + free_arg(arg->num.left); + free_arg(arg->num.right); + break; + + case FILTER_ARG_EXP: + free_arg(arg->exp.left); + free_arg(arg->exp.right); + break; + + case FILTER_ARG_STR: + free(arg->str.val); + regfree(&arg->str.reg); + free(arg->str.buffer); + break; + + case FILTER_ARG_VALUE: + if (arg->value.type == FILTER_STRING || + arg->value.type == FILTER_CHAR) + free(arg->value.str); + break; + + case FILTER_ARG_OP: + free_arg(arg->op.left); + free_arg(arg->op.right); + default: + break; + } + + free(arg); +} + +static void add_event(struct event_list **events, + struct event_format *event) +{ + struct event_list *list; + + list = malloc_or_die(sizeof(*list)); + list->next = *events; + *events = list; + list->event = event; +} + +static int event_match(struct event_format *event, + regex_t *sreg, regex_t *ereg) +{ + if (sreg) { + return !regexec(sreg, event->system, 0, NULL, 0) && + !regexec(ereg, event->name, 0, NULL, 0); + } + + return !regexec(ereg, event->system, 0, NULL, 0) || + !regexec(ereg, event->name, 0, NULL, 0); +} + +static int +find_event(struct pevent *pevent, struct event_list **events, + char *sys_name, char *event_name) +{ + struct event_format *event; + regex_t ereg; + regex_t sreg; + int match = 0; + char *reg; + int ret; + int i; + + if (!event_name) { + /* if no name is given, then swap sys and name */ + event_name = sys_name; + sys_name = NULL; + } + + reg = malloc_or_die(strlen(event_name) + 3); + sprintf(reg, "^%s$", event_name); + + ret = regcomp(&ereg, reg, REG_ICASE|REG_NOSUB); + free(reg); + + if (ret) + return -1; + + if (sys_name) { + reg = malloc_or_die(strlen(sys_name) + 3); + sprintf(reg, "^%s$", sys_name); + ret = regcomp(&sreg, reg, REG_ICASE|REG_NOSUB); + free(reg); + if (ret) { + regfree(&ereg); + return -1; + } + } + + for (i = 0; i < pevent->nr_events; i++) { + event = pevent->events[i]; + if (event_match(event, sys_name ? &sreg : NULL, &ereg)) { + match = 1; + add_event(events, event); + } + } + + regfree(&ereg); + if (sys_name) + regfree(&sreg); + + if (!match) + return -1; + + return 0; +} + +static void free_events(struct event_list *events) +{ + struct event_list *event; + + while (events) { + event = events; + events = events->next; + free(event); + } +} + +static struct filter_arg * +create_arg_item(struct event_format *event, const char *token, + enum event_type type, char **error_str) +{ + struct format_field *field; + struct filter_arg *arg; + + arg = allocate_arg(); + + switch (type) { + + case EVENT_SQUOTE: + case EVENT_DQUOTE: + arg->type = FILTER_ARG_VALUE; + arg->value.type = + type == EVENT_DQUOTE ? FILTER_STRING : FILTER_CHAR; + arg->value.str = strdup(token); + if (!arg->value.str) + die("malloc string"); + break; + case EVENT_ITEM: + /* if it is a number, then convert it */ + if (isdigit(token[0])) { + arg->type = FILTER_ARG_VALUE; + arg->value.type = FILTER_NUMBER; + arg->value.val = strtoull(token, NULL, 0); + break; + } + /* Consider this a field */ + field = pevent_find_any_field(event, token); + if (!field) { + if (strcmp(token, COMM) != 0) { + /* not a field, Make it false */ + arg->type = FILTER_ARG_BOOLEAN; + arg->boolean.value = FILTER_FALSE; + break; + } + /* If token is 'COMM' then it is special */ + field = &comm; + } + arg->type = FILTER_ARG_FIELD; + arg->field.field = field; + break; + default: + free_arg(arg); + show_error(error_str, "expected a value but found %s", + token); + return NULL; + } + return arg; +} + +static struct filter_arg * +create_arg_op(enum filter_op_type btype) +{ + struct filter_arg *arg; + + arg = allocate_arg(); + arg->type = FILTER_ARG_OP; + arg->op.type = btype; + + return arg; +} + +static struct filter_arg * +create_arg_exp(enum filter_exp_type etype) +{ + struct filter_arg *arg; + + arg = allocate_arg(); + arg->type = FILTER_ARG_EXP; + arg->op.type = etype; + + return arg; +} + +static struct filter_arg * +create_arg_cmp(enum filter_exp_type etype) +{ + struct filter_arg *arg; + + arg = allocate_arg(); + /* Use NUM and change if necessary */ + arg->type = FILTER_ARG_NUM; + arg->op.type = etype; + + return arg; +} + +static int add_right(struct filter_arg *op, struct filter_arg *arg, + char **error_str) +{ + struct filter_arg *left; + char *str; + int op_type; + int ret; + + switch (op->type) { + case FILTER_ARG_EXP: + if (op->exp.right) + goto out_fail; + op->exp.right = arg; + break; + + case FILTER_ARG_OP: + if (op->op.right) + goto out_fail; + op->op.right = arg; + break; + + case FILTER_ARG_NUM: + if (op->op.right) + goto out_fail; + /* + * The arg must be num, str, or field + */ + switch (arg->type) { + case FILTER_ARG_VALUE: + case FILTER_ARG_FIELD: + break; + default: + show_error(error_str, + "Illegal rvalue"); + return -1; + } + + /* + * Depending on the type, we may need to + * convert this to a string or regex. + */ + switch (arg->value.type) { + case FILTER_CHAR: + /* + * A char should be converted to number if + * the string is 1 byte, and the compare + * is not a REGEX. + */ + if (strlen(arg->value.str) == 1 && + op->num.type != FILTER_CMP_REGEX && + op->num.type != FILTER_CMP_NOT_REGEX) { + arg->value.type = FILTER_NUMBER; + goto do_int; + } + /* fall through */ + case FILTER_STRING: + + /* convert op to a string arg */ + op_type = op->num.type; + left = op->num.left; + str = arg->value.str; + + /* reset the op for the new field */ + memset(op, 0, sizeof(*op)); + + /* + * If left arg was a field not found then + * NULL the entire op. + */ + if (left->type == FILTER_ARG_BOOLEAN) { + free_arg(left); + free_arg(arg); + op->type = FILTER_ARG_BOOLEAN; + op->boolean.value = FILTER_FALSE; + break; + } + + /* Left arg must be a field */ + if (left->type != FILTER_ARG_FIELD) { + show_error(error_str, + "Illegal lvalue for string comparison"); + return -1; + } + + /* Make sure this is a valid string compare */ + switch (op_type) { + case FILTER_CMP_EQ: + op_type = FILTER_CMP_MATCH; + break; + case FILTER_CMP_NE: + op_type = FILTER_CMP_NOT_MATCH; + break; + + case FILTER_CMP_REGEX: + case FILTER_CMP_NOT_REGEX: + ret = regcomp(&op->str.reg, str, REG_ICASE|REG_NOSUB); + if (ret) { + show_error(error_str, + "RegEx '%s' did not compute", + str); + return -1; + } + break; + default: + show_error(error_str, + "Illegal comparison for string"); + return -1; + } + + op->type = FILTER_ARG_STR; + op->str.type = op_type; + op->str.field = left->field.field; + op->str.val = strdup(str); + if (!op->str.val) + die("malloc string"); + /* + * Need a buffer to copy data for tests + */ + op->str.buffer = malloc_or_die(op->str.field->size + 1); + /* Null terminate this buffer */ + op->str.buffer[op->str.field->size] = 0; + + /* We no longer have left or right args */ + free_arg(arg); + free_arg(left); + + break; + + case FILTER_NUMBER: + + do_int: + switch (op->num.type) { + case FILTER_CMP_REGEX: + case FILTER_CMP_NOT_REGEX: + show_error(error_str, + "Op not allowed with integers"); + return -1; + + default: + break; + } + + /* numeric compare */ + op->num.right = arg; + break; + default: + goto out_fail; + } + break; + default: + goto out_fail; + } + + return 0; + + out_fail: + show_error(error_str, + "Syntax error"); + return -1; +} + +static struct filter_arg * +rotate_op_right(struct filter_arg *a, struct filter_arg *b) +{ + struct filter_arg *arg; + + arg = a->op.right; + a->op.right = b; + return arg; +} + +static int add_left(struct filter_arg *op, struct filter_arg *arg) +{ + switch (op->type) { + case FILTER_ARG_EXP: + if (arg->type == FILTER_ARG_OP) + arg = rotate_op_right(arg, op); + op->exp.left = arg; + break; + + case FILTER_ARG_OP: + op->op.left = arg; + break; + case FILTER_ARG_NUM: + if (arg->type == FILTER_ARG_OP) + arg = rotate_op_right(arg, op); + + /* left arg of compares must be a field */ + if (arg->type != FILTER_ARG_FIELD && + arg->type != FILTER_ARG_BOOLEAN) + return -1; + op->num.left = arg; + break; + default: + return -1; + } + return 0; +} + +enum op_type { + OP_NONE, + OP_BOOL, + OP_NOT, + OP_EXP, + OP_CMP, +}; + +static enum op_type process_op(const char *token, + enum filter_op_type *btype, + enum filter_cmp_type *ctype, + enum filter_exp_type *etype) +{ + *btype = FILTER_OP_NOT; + *etype = FILTER_EXP_NONE; + *ctype = FILTER_CMP_NONE; + + if (strcmp(token, "&&") == 0) + *btype = FILTER_OP_AND; + else if (strcmp(token, "||") == 0) + *btype = FILTER_OP_OR; + else if (strcmp(token, "!") == 0) + return OP_NOT; + + if (*btype != FILTER_OP_NOT) + return OP_BOOL; + + /* Check for value expressions */ + if (strcmp(token, "+") == 0) { + *etype = FILTER_EXP_ADD; + } else if (strcmp(token, "-") == 0) { + *etype = FILTER_EXP_SUB; + } else if (strcmp(token, "*") == 0) { + *etype = FILTER_EXP_MUL; + } else if (strcmp(token, "/") == 0) { + *etype = FILTER_EXP_DIV; + } else if (strcmp(token, "%") == 0) { + *etype = FILTER_EXP_MOD; + } else if (strcmp(token, ">>") == 0) { + *etype = FILTER_EXP_RSHIFT; + } else if (strcmp(token, "<<") == 0) { + *etype = FILTER_EXP_LSHIFT; + } else if (strcmp(token, "&") == 0) { + *etype = FILTER_EXP_AND; + } else if (strcmp(token, "|") == 0) { + *etype = FILTER_EXP_OR; + } else if (strcmp(token, "^") == 0) { + *etype = FILTER_EXP_XOR; + } else if (strcmp(token, "~") == 0) + *etype = FILTER_EXP_NOT; + + if (*etype != FILTER_EXP_NONE) + return OP_EXP; + + /* Check for compares */ + if (strcmp(token, "==") == 0) + *ctype = FILTER_CMP_EQ; + else if (strcmp(token, "!=") == 0) + *ctype = FILTER_CMP_NE; + else if (strcmp(token, "<") == 0) + *ctype = FILTER_CMP_LT; + else if (strcmp(token, ">") == 0) + *ctype = FILTER_CMP_GT; + else if (strcmp(token, "<=") == 0) + *ctype = FILTER_CMP_LE; + else if (strcmp(token, ">=") == 0) + *ctype = FILTER_CMP_GE; + else if (strcmp(token, "=~") == 0) + *ctype = FILTER_CMP_REGEX; + else if (strcmp(token, "!~") == 0) + *ctype = FILTER_CMP_NOT_REGEX; + else + return OP_NONE; + + return OP_CMP; +} + +static int check_op_done(struct filter_arg *arg) +{ + switch (arg->type) { + case FILTER_ARG_EXP: + return arg->exp.right != NULL; + + case FILTER_ARG_OP: + return arg->op.right != NULL; + + case FILTER_ARG_NUM: + return arg->num.right != NULL; + + case FILTER_ARG_STR: + /* A string conversion is always done */ + return 1; + + case FILTER_ARG_BOOLEAN: + /* field not found, is ok */ + return 1; + + default: + return 0; + } +} + +enum filter_vals { + FILTER_VAL_NORM, + FILTER_VAL_FALSE, + FILTER_VAL_TRUE, +}; + +void reparent_op_arg(struct filter_arg *parent, struct filter_arg *old_child, + struct filter_arg *arg) +{ + struct filter_arg *other_child; + struct filter_arg **ptr; + + if (parent->type != FILTER_ARG_OP && + arg->type != FILTER_ARG_OP) + die("can not reparent other than OP"); + + /* Get the sibling */ + if (old_child->op.right == arg) { + ptr = &old_child->op.right; + other_child = old_child->op.left; + } else if (old_child->op.left == arg) { + ptr = &old_child->op.left; + other_child = old_child->op.right; + } else + die("Error in reparent op, find other child"); + + /* Detach arg from old_child */ + *ptr = NULL; + + /* Check for root */ + if (parent == old_child) { + free_arg(other_child); + *parent = *arg; + /* Free arg without recussion */ + free(arg); + return; + } + + if (parent->op.right == old_child) + ptr = &parent->op.right; + else if (parent->op.left == old_child) + ptr = &parent->op.left; + else + die("Error in reparent op"); + *ptr = arg; + + free_arg(old_child); +} + +enum filter_vals test_arg(struct filter_arg *parent, struct filter_arg *arg) +{ + enum filter_vals lval, rval; + + switch (arg->type) { + + /* bad case */ + case FILTER_ARG_BOOLEAN: + return FILTER_VAL_FALSE + arg->boolean.value; + + /* good cases: */ + case FILTER_ARG_STR: + case FILTER_ARG_VALUE: + case FILTER_ARG_FIELD: + return FILTER_VAL_NORM; + + case FILTER_ARG_EXP: + lval = test_arg(arg, arg->exp.left); + if (lval != FILTER_VAL_NORM) + return lval; + rval = test_arg(arg, arg->exp.right); + if (rval != FILTER_VAL_NORM) + return rval; + return FILTER_VAL_NORM; + + case FILTER_ARG_NUM: + lval = test_arg(arg, arg->num.left); + if (lval != FILTER_VAL_NORM) + return lval; + rval = test_arg(arg, arg->num.right); + if (rval != FILTER_VAL_NORM) + return rval; + return FILTER_VAL_NORM; + + case FILTER_ARG_OP: + if (arg->op.type != FILTER_OP_NOT) { + lval = test_arg(arg, arg->op.left); + switch (lval) { + case FILTER_VAL_NORM: + break; + case FILTER_VAL_TRUE: + if (arg->op.type == FILTER_OP_OR) + return FILTER_VAL_TRUE; + rval = test_arg(arg, arg->op.right); + if (rval != FILTER_VAL_NORM) + return rval; + + reparent_op_arg(parent, arg, arg->op.right); + return FILTER_VAL_NORM; + + case FILTER_VAL_FALSE: + if (arg->op.type == FILTER_OP_AND) + return FILTER_VAL_FALSE; + rval = test_arg(arg, arg->op.right); + if (rval != FILTER_VAL_NORM) + return rval; + + reparent_op_arg(parent, arg, arg->op.right); + return FILTER_VAL_NORM; + } + } + + rval = test_arg(arg, arg->op.right); + switch (rval) { + case FILTER_VAL_NORM: + break; + case FILTER_VAL_TRUE: + if (arg->op.type == FILTER_OP_OR) + return FILTER_VAL_TRUE; + if (arg->op.type == FILTER_OP_NOT) + return FILTER_VAL_FALSE; + + reparent_op_arg(parent, arg, arg->op.left); + return FILTER_VAL_NORM; + + case FILTER_VAL_FALSE: + if (arg->op.type == FILTER_OP_AND) + return FILTER_VAL_FALSE; + if (arg->op.type == FILTER_OP_NOT) + return FILTER_VAL_TRUE; + + reparent_op_arg(parent, arg, arg->op.left); + return FILTER_VAL_NORM; + } + + return FILTER_VAL_NORM; + default: + die("bad arg in filter tree"); + } + return FILTER_VAL_NORM; +} + +/* Remove any unknown event fields */ +static struct filter_arg *collapse_tree(struct filter_arg *arg) +{ + enum filter_vals ret; + + ret = test_arg(arg, arg); + switch (ret) { + case FILTER_VAL_NORM: + return arg; + + case FILTER_VAL_TRUE: + case FILTER_VAL_FALSE: + free_arg(arg); + arg = allocate_arg(); + arg->type = FILTER_ARG_BOOLEAN; + arg->boolean.value = ret == FILTER_VAL_TRUE; + } + + return arg; +} + +static int +process_filter(struct event_format *event, struct filter_arg **parg, + char **error_str, int not) +{ + enum event_type type; + char *token = NULL; + struct filter_arg *current_op = NULL; + struct filter_arg *current_exp = NULL; + struct filter_arg *left_item = NULL; + struct filter_arg *arg = NULL; + enum op_type op_type; + enum filter_op_type btype; + enum filter_exp_type etype; + enum filter_cmp_type ctype; + int ret; + + *parg = NULL; + + do { + free(token); + type = read_token(&token); + switch (type) { + case EVENT_SQUOTE: + case EVENT_DQUOTE: + case EVENT_ITEM: + arg = create_arg_item(event, token, type, error_str); + if (!arg) + goto fail; + if (!left_item) + left_item = arg; + else if (current_exp) { + ret = add_right(current_exp, arg, error_str); + if (ret < 0) + goto fail; + left_item = NULL; + /* Not's only one one expression */ + if (not) { + arg = NULL; + if (current_op) + goto fail_print; + free(token); + *parg = current_exp; + return 0; + } + } else + goto fail_print; + arg = NULL; + break; + + case EVENT_DELIM: + if (*token == ',') { + show_error(error_str, + "Illegal token ','"); + goto fail; + } + + if (*token == '(') { + if (left_item) { + show_error(error_str, + "Open paren can not come after item"); + goto fail; + } + if (current_exp) { + show_error(error_str, + "Open paren can not come after expression"); + goto fail; + } + + ret = process_filter(event, &arg, error_str, 0); + if (ret != 1) { + if (ret == 0) + show_error(error_str, + "Unbalanced number of '('"); + goto fail; + } + ret = 0; + + /* A not wants just one expression */ + if (not) { + if (current_op) + goto fail_print; + *parg = arg; + return 0; + } + + if (current_op) + ret = add_right(current_op, arg, error_str); + else + current_exp = arg; + + if (ret < 0) + goto fail; + + } else { /* ')' */ + if (!current_op && !current_exp) + goto fail_print; + + /* Make sure everything is finished at this level */ + if (current_exp && !check_op_done(current_exp)) + goto fail_print; + if (current_op && !check_op_done(current_op)) + goto fail_print; + + if (current_op) + *parg = current_op; + else + *parg = current_exp; + return 1; + } + break; + + case EVENT_OP: + op_type = process_op(token, &btype, &ctype, &etype); + + /* All expect a left arg except for NOT */ + switch (op_type) { + case OP_BOOL: + /* Logic ops need a left expression */ + if (!current_exp && !current_op) + goto fail_print; + /* fall through */ + case OP_NOT: + /* logic only processes ops and exp */ + if (left_item) + goto fail_print; + break; + case OP_EXP: + case OP_CMP: + if (!left_item) + goto fail_print; + break; + case OP_NONE: + show_error(error_str, + "Unknown op token %s", token); + goto fail; + } + + ret = 0; + switch (op_type) { + case OP_BOOL: + arg = create_arg_op(btype); + if (current_op) + ret = add_left(arg, current_op); + else + ret = add_left(arg, current_exp); + current_op = arg; + current_exp = NULL; + break; + + case OP_NOT: + arg = create_arg_op(btype); + if (current_op) + ret = add_right(current_op, arg, error_str); + if (ret < 0) + goto fail; + current_exp = arg; + ret = process_filter(event, &arg, error_str, 1); + if (ret < 0) + goto fail; + ret = add_right(current_exp, arg, error_str); + if (ret < 0) + goto fail; + break; + + case OP_EXP: + case OP_CMP: + if (op_type == OP_EXP) + arg = create_arg_exp(etype); + else + arg = create_arg_cmp(ctype); + + if (current_op) + ret = add_right(current_op, arg, error_str); + if (ret < 0) + goto fail; + ret = add_left(arg, left_item); + if (ret < 0) { + arg = NULL; + goto fail_print; + } + current_exp = arg; + break; + default: + break; + } + arg = NULL; + if (ret < 0) + goto fail_print; + break; + case EVENT_NONE: + break; + default: + goto fail_print; + } + } while (type != EVENT_NONE); + + if (!current_op && !current_exp) + goto fail_print; + + if (!current_op) + current_op = current_exp; + + current_op = collapse_tree(current_op); + + *parg = current_op; + + return 0; + + fail_print: + show_error(error_str, "Syntax error"); + fail: + free_arg(current_op); + free_arg(current_exp); + free_arg(arg); + free(token); + return -1; +} + +static int +process_event(struct event_format *event, const char *filter_str, + struct filter_arg **parg, char **error_str) +{ + int ret; + + pevent_buffer_init(filter_str, strlen(filter_str)); + + ret = process_filter(event, parg, error_str, 0); + if (ret == 1) { + show_error(error_str, + "Unbalanced number of ')'"); + return -1; + } + if (ret < 0) + return ret; + + /* If parg is NULL, then make it into FALSE */ + if (!*parg) { + *parg = allocate_arg(); + (*parg)->type = FILTER_ARG_BOOLEAN; + (*parg)->boolean.value = FILTER_FALSE; + } + + return 0; +} + +static int filter_event(struct event_filter *filter, + struct event_format *event, + const char *filter_str, char **error_str) +{ + struct filter_type *filter_type; + struct filter_arg *arg; + int ret; + + if (filter_str) { + ret = process_event(event, filter_str, &arg, error_str); + if (ret < 0) + return ret; + + } else { + /* just add a TRUE arg */ + arg = allocate_arg(); + arg->type = FILTER_ARG_BOOLEAN; + arg->boolean.value = FILTER_TRUE; + } + + filter_type = add_filter_type(filter, event->id); + if (filter_type->filter) + free_arg(filter_type->filter); + filter_type->filter = arg; + + return 0; +} + +/** + * pevent_filter_add_filter_str - add a new filter + * @filter: the event filter to add to + * @filter_str: the filter string that contains the filter + * @error_str: string containing reason for failed filter + * + * Returns 0 if the filter was successfully added + * -1 if there was an error. + * + * On error, if @error_str points to a string pointer, + * it is set to the reason that the filter failed. + * This string must be freed with "free". + */ +int pevent_filter_add_filter_str(struct event_filter *filter, + const char *filter_str, + char **error_str) +{ + struct pevent *pevent = filter->pevent; + struct event_list *event; + struct event_list *events = NULL; + const char *filter_start; + const char *next_event; + char *this_event; + char *event_name = NULL; + char *sys_name = NULL; + char *sp; + int rtn = 0; + int len; + int ret; + + /* clear buffer to reset show error */ + pevent_buffer_init("", 0); + + if (error_str) + *error_str = NULL; + + filter_start = strchr(filter_str, ':'); + if (filter_start) + len = filter_start - filter_str; + else + len = strlen(filter_str); + + + do { + next_event = strchr(filter_str, ','); + if (next_event && + (!filter_start || next_event < filter_start)) + len = next_event - filter_str; + else if (filter_start) + len = filter_start - filter_str; + else + len = strlen(filter_str); + + this_event = malloc_or_die(len + 1); + memcpy(this_event, filter_str, len); + this_event[len] = 0; + + if (next_event) + next_event++; + + filter_str = next_event; + + sys_name = strtok_r(this_event, "/", &sp); + event_name = strtok_r(NULL, "/", &sp); + + if (!sys_name) { + show_error(error_str, "No filter found"); + /* This can only happen when events is NULL, but still */ + free_events(events); + free(this_event); + return -1; + } + + /* Find this event */ + ret = find_event(pevent, &events, strim(sys_name), strim(event_name)); + if (ret < 0) { + if (event_name) + show_error(error_str, + "No event found under '%s.%s'", + sys_name, event_name); + else + show_error(error_str, + "No event found under '%s'", + sys_name); + free_events(events); + free(this_event); + return -1; + } + free(this_event); + } while (filter_str); + + /* Skip the ':' */ + if (filter_start) + filter_start++; + + /* filter starts here */ + for (event = events; event; event = event->next) { + ret = filter_event(filter, event->event, filter_start, + error_str); + /* Failures are returned if a parse error happened */ + if (ret < 0) + rtn = ret; + + if (ret >= 0 && pevent->test_filters) { + char *test; + test = pevent_filter_make_string(filter, event->event->id); + printf(" '%s: %s'\n", event->event->name, test); + free(test); + } + } + + free_events(events); + + if (rtn >= 0 && pevent->test_filters) + exit(0); + + return rtn; +} + +static void free_filter_type(struct filter_type *filter_type) +{ + free_arg(filter_type->filter); +} + +/** + * pevent_filter_remove_event - remove a filter for an event + * @filter: the event filter to remove from + * @event_id: the event to remove a filter for + * + * Removes the filter saved for an event defined by @event_id + * from the @filter. + * + * Returns 1: if an event was removed + * 0: if the event was not found + */ +int pevent_filter_remove_event(struct event_filter *filter, + int event_id) +{ + struct filter_type *filter_type; + unsigned long len; + + if (!filter->filters) + return 0; + + filter_type = find_filter_type(filter, event_id); + + if (!filter_type) + return 0; + + free_filter_type(filter_type); + + /* The filter_type points into the event_filters array */ + len = (unsigned long)(filter->event_filters + filter->filters) - + (unsigned long)(filter_type + 1); + + memmove(filter_type, filter_type + 1, len); + filter->filters--; + + memset(&filter->event_filters[filter->filters], 0, + sizeof(*filter_type)); + + return 1; +} + +/** + * pevent_filter_reset - clear all filters in a filter + * @filter: the event filter to reset + * + * Removes all filters from a filter and resets it. + */ +void pevent_filter_reset(struct event_filter *filter) +{ + int i; + + for (i = 0; i < filter->filters; i++) + free_filter_type(&filter->event_filters[i]); + + free(filter->event_filters); + filter->filters = 0; + filter->event_filters = NULL; +} + +void pevent_filter_free(struct event_filter *filter) +{ + pevent_unref(filter->pevent); + + pevent_filter_reset(filter); + + free(filter); +} + +static char *arg_to_str(struct event_filter *filter, struct filter_arg *arg); + +static int copy_filter_type(struct event_filter *filter, + struct event_filter *source, + struct filter_type *filter_type) +{ + struct filter_arg *arg; + struct event_format *event; + const char *sys; + const char *name; + char *str; + + /* Can't assume that the pevent's are the same */ + sys = filter_type->event->system; + name = filter_type->event->name; + event = pevent_find_event_by_name(filter->pevent, sys, name); + if (!event) + return -1; + + str = arg_to_str(source, filter_type->filter); + if (!str) + return -1; + + if (strcmp(str, "TRUE") == 0 || strcmp(str, "FALSE") == 0) { + /* Add trivial event */ + arg = allocate_arg(); + arg->type = FILTER_ARG_BOOLEAN; + if (strcmp(str, "TRUE") == 0) + arg->boolean.value = 1; + else + arg->boolean.value = 0; + + filter_type = add_filter_type(filter, event->id); + filter_type->filter = arg; + + free(str); + return 0; + } + + filter_event(filter, event, str, NULL); + free(str); + + return 0; +} + +/** + * pevent_filter_copy - copy a filter using another filter + * @dest - the filter to copy to + * @source - the filter to copy from + * + * Returns 0 on success and -1 if not all filters were copied + */ +int pevent_filter_copy(struct event_filter *dest, struct event_filter *source) +{ + int ret = 0; + int i; + + pevent_filter_reset(dest); + + for (i = 0; i < source->filters; i++) { + if (copy_filter_type(dest, source, &source->event_filters[i])) + ret = -1; + } + return ret; +} + + +/** + * pevent_update_trivial - update the trivial filters with the given filter + * @dest - the filter to update + * @source - the filter as the source of the update + * @type - the type of trivial filter to update. + * + * Scan dest for trivial events matching @type to replace with the source. + * + * Returns 0 on success and -1 if there was a problem updating, but + * events may have still been updated on error. + */ +int pevent_update_trivial(struct event_filter *dest, struct event_filter *source, + enum filter_trivial_type type) +{ + struct pevent *src_pevent; + struct pevent *dest_pevent; + struct event_format *event; + struct filter_type *filter_type; + struct filter_arg *arg; + char *str; + int i; + + src_pevent = source->pevent; + dest_pevent = dest->pevent; + + /* Do nothing if either of the filters has nothing to filter */ + if (!dest->filters || !source->filters) + return 0; + + for (i = 0; i < dest->filters; i++) { + filter_type = &dest->event_filters[i]; + arg = filter_type->filter; + if (arg->type != FILTER_ARG_BOOLEAN) + continue; + if ((arg->boolean.value && type == FILTER_TRIVIAL_FALSE) || + (!arg->boolean.value && type == FILTER_TRIVIAL_TRUE)) + continue; + + event = filter_type->event; + + if (src_pevent != dest_pevent) { + /* do a look up */ + event = pevent_find_event_by_name(src_pevent, + event->system, + event->name); + if (!event) + return -1; + } + + str = pevent_filter_make_string(source, event->id); + if (!str) + continue; + + /* Don't bother if the filter is trivial too */ + if (strcmp(str, "TRUE") != 0 && strcmp(str, "FALSE") != 0) + filter_event(dest, event, str, NULL); + free(str); + } + return 0; +} + +/** + * pevent_filter_clear_trivial - clear TRUE and FALSE filters + * @filter: the filter to remove trivial filters from + * @type: remove only true, false, or both + * + * Removes filters that only contain a TRUE or FALES boolean arg. + */ +void pevent_filter_clear_trivial(struct event_filter *filter, + enum filter_trivial_type type) +{ + struct filter_type *filter_type; + int count = 0; + int *ids = NULL; + int i; + + if (!filter->filters) + return; + + /* + * Two steps, first get all ids with trivial filters. + * then remove those ids. + */ + for (i = 0; i < filter->filters; i++) { + filter_type = &filter->event_filters[i]; + if (filter_type->filter->type != FILTER_ARG_BOOLEAN) + continue; + switch (type) { + case FILTER_TRIVIAL_FALSE: + if (filter_type->filter->boolean.value) + continue; + case FILTER_TRIVIAL_TRUE: + if (!filter_type->filter->boolean.value) + continue; + default: + break; + } + + ids = realloc(ids, sizeof(*ids) * (count + 1)); + if (!ids) + die("Can't allocate ids"); + ids[count++] = filter_type->event_id; + } + + if (!count) + return; + + for (i = 0; i < count; i++) + pevent_filter_remove_event(filter, ids[i]); + + free(ids); +} + +/** + * pevent_filter_event_has_trivial - return true event contains trivial filter + * @filter: the filter with the information + * @event_id: the id of the event to test + * @type: trivial type to test for (TRUE, FALSE, EITHER) + * + * Returns 1 if the event contains a matching trivial type + * otherwise 0. + */ +int pevent_filter_event_has_trivial(struct event_filter *filter, + int event_id, + enum filter_trivial_type type) +{ + struct filter_type *filter_type; + + if (!filter->filters) + return 0; + + filter_type = find_filter_type(filter, event_id); + + if (!filter_type) + return 0; + + if (filter_type->filter->type != FILTER_ARG_BOOLEAN) + return 0; + + switch (type) { + case FILTER_TRIVIAL_FALSE: + return !filter_type->filter->boolean.value; + + case FILTER_TRIVIAL_TRUE: + return filter_type->filter->boolean.value; + default: + return 1; + } +} + +static int test_filter(struct event_format *event, + struct filter_arg *arg, struct pevent_record *record); + +static const char * +get_comm(struct event_format *event, struct pevent_record *record) +{ + const char *comm; + int pid; + + pid = pevent_data_pid(event->pevent, record); + comm = pevent_data_comm_from_pid(event->pevent, pid); + return comm; +} + +static unsigned long long +get_value(struct event_format *event, + struct format_field *field, struct pevent_record *record) +{ + unsigned long long val; + + /* Handle our dummy "comm" field */ + if (field == &comm) { + const char *name; + + name = get_comm(event, record); + return (unsigned long)name; + } + + pevent_read_number_field(field, record->data, &val); + + if (!(field->flags & FIELD_IS_SIGNED)) + return val; + + switch (field->size) { + case 1: + return (char)val; + case 2: + return (short)val; + case 4: + return (int)val; + case 8: + return (long long)val; + } + return val; +} + +static unsigned long long +get_arg_value(struct event_format *event, struct filter_arg *arg, struct pevent_record *record); + +static unsigned long long +get_exp_value(struct event_format *event, struct filter_arg *arg, struct pevent_record *record) +{ + unsigned long long lval, rval; + + lval = get_arg_value(event, arg->exp.left, record); + rval = get_arg_value(event, arg->exp.right, record); + + switch (arg->exp.type) { + case FILTER_EXP_ADD: + return lval + rval; + + case FILTER_EXP_SUB: + return lval - rval; + + case FILTER_EXP_MUL: + return lval * rval; + + case FILTER_EXP_DIV: + return lval / rval; + + case FILTER_EXP_MOD: + return lval % rval; + + case FILTER_EXP_RSHIFT: + return lval >> rval; + + case FILTER_EXP_LSHIFT: + return lval << rval; + + case FILTER_EXP_AND: + return lval & rval; + + case FILTER_EXP_OR: + return lval | rval; + + case FILTER_EXP_XOR: + return lval ^ rval; + + case FILTER_EXP_NOT: + default: + die("error in exp"); + } + return 0; +} + +static unsigned long long +get_arg_value(struct event_format *event, struct filter_arg *arg, struct pevent_record *record) +{ + switch (arg->type) { + case FILTER_ARG_FIELD: + return get_value(event, arg->field.field, record); + + case FILTER_ARG_VALUE: + if (arg->value.type != FILTER_NUMBER) + die("must have number field!"); + return arg->value.val; + + case FILTER_ARG_EXP: + return get_exp_value(event, arg, record); + + default: + die("oops in filter"); + } + return 0; +} + +static int test_num(struct event_format *event, + struct filter_arg *arg, struct pevent_record *record) +{ + unsigned long long lval, rval; + + lval = get_arg_value(event, arg->num.left, record); + rval = get_arg_value(event, arg->num.right, record); + + switch (arg->num.type) { + case FILTER_CMP_EQ: + return lval == rval; + + case FILTER_CMP_NE: + return lval != rval; + + case FILTER_CMP_GT: + return lval > rval; + + case FILTER_CMP_LT: + return lval < rval; + + case FILTER_CMP_GE: + return lval >= rval; + + case FILTER_CMP_LE: + return lval <= rval; + + default: + /* ?? */ + return 0; + } +} + +static const char *get_field_str(struct filter_arg *arg, struct pevent_record *record) +{ + struct event_format *event; + struct pevent *pevent; + unsigned long long addr; + const char *val = NULL; + char hex[64]; + + /* If the field is not a string convert it */ + if (arg->str.field->flags & FIELD_IS_STRING) { + val = record->data + arg->str.field->offset; + + /* + * We need to copy the data since we can't be sure the field + * is null terminated. + */ + if (*(val + arg->str.field->size - 1)) { + /* copy it */ + memcpy(arg->str.buffer, val, arg->str.field->size); + /* the buffer is already NULL terminated */ + val = arg->str.buffer; + } + + } else { + event = arg->str.field->event; + pevent = event->pevent; + addr = get_value(event, arg->str.field, record); + + if (arg->str.field->flags & (FIELD_IS_POINTER | FIELD_IS_LONG)) + /* convert to a kernel symbol */ + val = pevent_find_function(pevent, addr); + + if (val == NULL) { + /* just use the hex of the string name */ + snprintf(hex, 64, "0x%llx", addr); + val = hex; + } + } + + return val; +} + +static int test_str(struct event_format *event, + struct filter_arg *arg, struct pevent_record *record) +{ + const char *val; + + if (arg->str.field == &comm) + val = get_comm(event, record); + else + val = get_field_str(arg, record); + + switch (arg->str.type) { + case FILTER_CMP_MATCH: + return strcmp(val, arg->str.val) == 0; + + case FILTER_CMP_NOT_MATCH: + return strcmp(val, arg->str.val) != 0; + + case FILTER_CMP_REGEX: + /* Returns zero on match */ + return !regexec(&arg->str.reg, val, 0, NULL, 0); + + case FILTER_CMP_NOT_REGEX: + return regexec(&arg->str.reg, val, 0, NULL, 0); + + default: + /* ?? */ + return 0; + } +} + +static int test_op(struct event_format *event, + struct filter_arg *arg, struct pevent_record *record) +{ + switch (arg->op.type) { + case FILTER_OP_AND: + return test_filter(event, arg->op.left, record) && + test_filter(event, arg->op.right, record); + + case FILTER_OP_OR: + return test_filter(event, arg->op.left, record) || + test_filter(event, arg->op.right, record); + + case FILTER_OP_NOT: + return !test_filter(event, arg->op.right, record); + + default: + /* ?? */ + return 0; + } +} + +static int test_filter(struct event_format *event, + struct filter_arg *arg, struct pevent_record *record) +{ + switch (arg->type) { + case FILTER_ARG_BOOLEAN: + /* easy case */ + return arg->boolean.value; + + case FILTER_ARG_OP: + return test_op(event, arg, record); + + case FILTER_ARG_NUM: + return test_num(event, arg, record); + + case FILTER_ARG_STR: + return test_str(event, arg, record); + + case FILTER_ARG_EXP: + case FILTER_ARG_VALUE: + case FILTER_ARG_FIELD: + /* + * Expressions, fields and values evaluate + * to true if they return non zero + */ + return !!get_arg_value(event, arg, record); + + default: + die("oops!"); + /* ?? */ + return 0; + } +} + +/** + * pevent_event_filtered - return true if event has filter + * @filter: filter struct with filter information + * @event_id: event id to test if filter exists + * + * Returns 1 if filter found for @event_id + * otherwise 0; + */ +int pevent_event_filtered(struct event_filter *filter, + int event_id) +{ + struct filter_type *filter_type; + + if (!filter->filters) + return 0; + + filter_type = find_filter_type(filter, event_id); + + return filter_type ? 1 : 0; +} + +/** + * pevent_filter_match - test if a record matches a filter + * @filter: filter struct with filter information + * @record: the record to test against the filter + * + * Returns: + * 1 - filter found for event and @record matches + * 0 - filter found for event and @record does not match + * -1 - no filter found for @record's event + * -2 - if no filters exist + */ +int pevent_filter_match(struct event_filter *filter, + struct pevent_record *record) +{ + struct pevent *pevent = filter->pevent; + struct filter_type *filter_type; + int event_id; + + if (!filter->filters) + return FILTER_NONE; + + event_id = pevent_data_type(pevent, record); + + filter_type = find_filter_type(filter, event_id); + + if (!filter_type) + return FILTER_NOEXIST; + + return test_filter(filter_type->event, filter_type->filter, record) ? + FILTER_MATCH : FILTER_MISS; +} + +static char *op_to_str(struct event_filter *filter, struct filter_arg *arg) +{ + char *str = NULL; + char *left = NULL; + char *right = NULL; + char *op = NULL; + int left_val = -1; + int right_val = -1; + int val; + int len; + + switch (arg->op.type) { + case FILTER_OP_AND: + op = "&&"; + /* fall through */ + case FILTER_OP_OR: + if (!op) + op = "||"; + + left = arg_to_str(filter, arg->op.left); + right = arg_to_str(filter, arg->op.right); + if (!left || !right) + break; + + /* Try to consolidate boolean values */ + if (strcmp(left, "TRUE") == 0) + left_val = 1; + else if (strcmp(left, "FALSE") == 0) + left_val = 0; + + if (strcmp(right, "TRUE") == 0) + right_val = 1; + else if (strcmp(right, "FALSE") == 0) + right_val = 0; + + if (left_val >= 0) { + if ((arg->op.type == FILTER_OP_AND && !left_val) || + (arg->op.type == FILTER_OP_OR && left_val)) { + /* Just return left value */ + str = left; + left = NULL; + break; + } + if (right_val >= 0) { + /* just evaluate this. */ + val = 0; + switch (arg->op.type) { + case FILTER_OP_AND: + val = left_val && right_val; + break; + case FILTER_OP_OR: + val = left_val || right_val; + break; + default: + break; + } + str = malloc_or_die(6); + if (val) + strcpy(str, "TRUE"); + else + strcpy(str, "FALSE"); + break; + } + } + if (right_val >= 0) { + if ((arg->op.type == FILTER_OP_AND && !right_val) || + (arg->op.type == FILTER_OP_OR && right_val)) { + /* Just return right value */ + str = right; + right = NULL; + break; + } + /* The right value is meaningless */ + str = left; + left = NULL; + break; + } + + len = strlen(left) + strlen(right) + strlen(op) + 10; + str = malloc_or_die(len); + snprintf(str, len, "(%s) %s (%s)", + left, op, right); + break; + + case FILTER_OP_NOT: + op = "!"; + right = arg_to_str(filter, arg->op.right); + if (!right) + break; + + /* See if we can consolidate */ + if (strcmp(right, "TRUE") == 0) + right_val = 1; + else if (strcmp(right, "FALSE") == 0) + right_val = 0; + if (right_val >= 0) { + /* just return the opposite */ + str = malloc_or_die(6); + if (right_val) + strcpy(str, "FALSE"); + else + strcpy(str, "TRUE"); + break; + } + len = strlen(right) + strlen(op) + 3; + str = malloc_or_die(len); + snprintf(str, len, "%s(%s)", op, right); + break; + + default: + /* ?? */ + break; + } + free(left); + free(right); + return str; +} + +static char *val_to_str(struct event_filter *filter, struct filter_arg *arg) +{ + char *str; + + str = malloc_or_die(30); + + snprintf(str, 30, "%lld", arg->value.val); + + return str; +} + +static char *field_to_str(struct event_filter *filter, struct filter_arg *arg) +{ + return strdup(arg->field.field->name); +} + +static char *exp_to_str(struct event_filter *filter, struct filter_arg *arg) +{ + char *lstr; + char *rstr; + char *op; + char *str = NULL; + int len; + + lstr = arg_to_str(filter, arg->exp.left); + rstr = arg_to_str(filter, arg->exp.right); + if (!lstr || !rstr) + goto out; + + switch (arg->exp.type) { + case FILTER_EXP_ADD: + op = "+"; + break; + case FILTER_EXP_SUB: + op = "-"; + break; + case FILTER_EXP_MUL: + op = "*"; + break; + case FILTER_EXP_DIV: + op = "/"; + break; + case FILTER_EXP_MOD: + op = "%"; + break; + case FILTER_EXP_RSHIFT: + op = ">>"; + break; + case FILTER_EXP_LSHIFT: + op = "<<"; + break; + case FILTER_EXP_AND: + op = "&"; + break; + case FILTER_EXP_OR: + op = "|"; + break; + case FILTER_EXP_XOR: + op = "^"; + break; + default: + die("oops in exp"); + } + + len = strlen(op) + strlen(lstr) + strlen(rstr) + 4; + str = malloc_or_die(len); + snprintf(str, len, "%s %s %s", lstr, op, rstr); +out: + free(lstr); + free(rstr); + + return str; +} + +static char *num_to_str(struct event_filter *filter, struct filter_arg *arg) +{ + char *lstr; + char *rstr; + char *str = NULL; + char *op = NULL; + int len; + + lstr = arg_to_str(filter, arg->num.left); + rstr = arg_to_str(filter, arg->num.right); + if (!lstr || !rstr) + goto out; + + switch (arg->num.type) { + case FILTER_CMP_EQ: + op = "=="; + /* fall through */ + case FILTER_CMP_NE: + if (!op) + op = "!="; + /* fall through */ + case FILTER_CMP_GT: + if (!op) + op = ">"; + /* fall through */ + case FILTER_CMP_LT: + if (!op) + op = "<"; + /* fall through */ + case FILTER_CMP_GE: + if (!op) + op = ">="; + /* fall through */ + case FILTER_CMP_LE: + if (!op) + op = "<="; + + len = strlen(lstr) + strlen(op) + strlen(rstr) + 4; + str = malloc_or_die(len); + sprintf(str, "%s %s %s", lstr, op, rstr); + + break; + + default: + /* ?? */ + break; + } + +out: + free(lstr); + free(rstr); + return str; +} + +static char *str_to_str(struct event_filter *filter, struct filter_arg *arg) +{ + char *str = NULL; + char *op = NULL; + int len; + + switch (arg->str.type) { + case FILTER_CMP_MATCH: + op = "=="; + /* fall through */ + case FILTER_CMP_NOT_MATCH: + if (!op) + op = "!="; + /* fall through */ + case FILTER_CMP_REGEX: + if (!op) + op = "=~"; + /* fall through */ + case FILTER_CMP_NOT_REGEX: + if (!op) + op = "!~"; + + len = strlen(arg->str.field->name) + strlen(op) + + strlen(arg->str.val) + 6; + str = malloc_or_die(len); + snprintf(str, len, "%s %s \"%s\"", + arg->str.field->name, + op, arg->str.val); + break; + + default: + /* ?? */ + break; + } + return str; +} + +static char *arg_to_str(struct event_filter *filter, struct filter_arg *arg) +{ + char *str; + + switch (arg->type) { + case FILTER_ARG_BOOLEAN: + str = malloc_or_die(6); + if (arg->boolean.value) + strcpy(str, "TRUE"); + else + strcpy(str, "FALSE"); + return str; + + case FILTER_ARG_OP: + return op_to_str(filter, arg); + + case FILTER_ARG_NUM: + return num_to_str(filter, arg); + + case FILTER_ARG_STR: + return str_to_str(filter, arg); + + case FILTER_ARG_VALUE: + return val_to_str(filter, arg); + + case FILTER_ARG_FIELD: + return field_to_str(filter, arg); + + case FILTER_ARG_EXP: + return exp_to_str(filter, arg); + + default: + /* ?? */ + return NULL; + } + +} + +/** + * pevent_filter_make_string - return a string showing the filter + * @filter: filter struct with filter information + * @event_id: the event id to return the filter string with + * + * Returns a string that displays the filter contents. + * This string must be freed with free(str). + * NULL is returned if no filter is found. + */ +char * +pevent_filter_make_string(struct event_filter *filter, int event_id) +{ + struct filter_type *filter_type; + + if (!filter->filters) + return NULL; + + filter_type = find_filter_type(filter, event_id); + + if (!filter_type) + return NULL; + + return arg_to_str(filter, filter_type->filter); +} + +/** + * pevent_filter_compare - compare two filters and return if they are the same + * @filter1: Filter to compare with @filter2 + * @filter2: Filter to compare with @filter1 + * + * Returns: + * 1 if the two filters hold the same content. + * 0 if they do not. + */ +int pevent_filter_compare(struct event_filter *filter1, struct event_filter *filter2) +{ + struct filter_type *filter_type1; + struct filter_type *filter_type2; + char *str1, *str2; + int result; + int i; + + /* Do the easy checks first */ + if (filter1->filters != filter2->filters) + return 0; + if (!filter1->filters && !filter2->filters) + return 1; + + /* + * Now take a look at each of the events to see if they have the same + * filters to them. + */ + for (i = 0; i < filter1->filters; i++) { + filter_type1 = &filter1->event_filters[i]; + filter_type2 = find_filter_type(filter2, filter_type1->event_id); + if (!filter_type2) + break; + if (filter_type1->filter->type != filter_type2->filter->type) + break; + switch (filter_type1->filter->type) { + case FILTER_TRIVIAL_FALSE: + case FILTER_TRIVIAL_TRUE: + /* trivial types just need the type compared */ + continue; + default: + break; + } + /* The best way to compare complex filters is with strings */ + str1 = arg_to_str(filter1, filter_type1->filter); + str2 = arg_to_str(filter2, filter_type2->filter); + if (str1 && str2) + result = strcmp(str1, str2) != 0; + else + /* bail out if allocation fails */ + result = 1; + + free(str1); + free(str2); + if (result) + break; + } + + if (i < filter1->filters) + return 0; + return 1; +} + diff --git a/tools/lib/traceevent/parse-utils.c b/tools/lib/traceevent/parse-utils.c new file mode 100644 index 000000000..bba701cf1 --- /dev/null +++ b/tools/lib/traceevent/parse-utils.c @@ -0,0 +1,129 @@ +/* + * Copyright (C) 2010 Red Hat Inc, Steven Rostedt + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License (not later!) + * + * 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this program; if not, see + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + */ +#include +#include +#include +#include +#include + +#define __weak __attribute__((weak)) + +void __vdie(const char *fmt, va_list ap) +{ + int ret = errno; + + if (errno) + perror("trace-cmd"); + else + ret = -1; + + fprintf(stderr, " "); + vfprintf(stderr, fmt, ap); + + fprintf(stderr, "\n"); + exit(ret); +} + +void __die(const char *fmt, ...) +{ + va_list ap; + + va_start(ap, fmt); + __vdie(fmt, ap); + va_end(ap); +} + +void __weak die(const char *fmt, ...) +{ + va_list ap; + + va_start(ap, fmt); + __vdie(fmt, ap); + va_end(ap); +} + +void __vwarning(const char *fmt, va_list ap) +{ + if (errno) + perror("trace-cmd"); + errno = 0; + + fprintf(stderr, " "); + vfprintf(stderr, fmt, ap); + + fprintf(stderr, "\n"); +} + +void __warning(const char *fmt, ...) +{ + va_list ap; + + va_start(ap, fmt); + __vwarning(fmt, ap); + va_end(ap); +} + +void __weak warning(const char *fmt, ...) +{ + va_list ap; + + va_start(ap, fmt); + __vwarning(fmt, ap); + va_end(ap); +} + +void __vpr_stat(const char *fmt, va_list ap) +{ + vprintf(fmt, ap); + printf("\n"); +} + +void __pr_stat(const char *fmt, ...) +{ + va_list ap; + + va_start(ap, fmt); + __vpr_stat(fmt, ap); + va_end(ap); +} + +void __weak vpr_stat(const char *fmt, va_list ap) +{ + __vpr_stat(fmt, ap); +} + +void __weak pr_stat(const char *fmt, ...) +{ + va_list ap; + + va_start(ap, fmt); + __vpr_stat(fmt, ap); + va_end(ap); +} + +void __weak *malloc_or_die(unsigned int size) +{ + void *data; + + data = malloc(size); + if (!data) + die("malloc"); + return data; +} diff --git a/tools/lib/traceevent/trace-seq.c b/tools/lib/traceevent/trace-seq.c new file mode 100644 index 000000000..a57db8051 --- /dev/null +++ b/tools/lib/traceevent/trace-seq.c @@ -0,0 +1,199 @@ +/* + * Copyright (C) 2009 Red Hat Inc, Steven Rostedt + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License (not later!) + * + * 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this program; if not, see + * + * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + */ +#include +#include +#include +#include + +#include "event-parse.h" +#include "event-utils.h" + +/* + * The TRACE_SEQ_POISON is to catch the use of using + * a trace_seq structure after it was destroyed. + */ +#define TRACE_SEQ_POISON ((void *)0xdeadbeef) +#define TRACE_SEQ_CHECK(s) \ +do { \ + if ((s)->buffer == TRACE_SEQ_POISON) \ + die("Usage of trace_seq after it was destroyed"); \ +} while (0) + +/** + * trace_seq_init - initialize the trace_seq structure + * @s: a pointer to the trace_seq structure to initialize + */ +void trace_seq_init(struct trace_seq *s) +{ + s->len = 0; + s->readpos = 0; + s->buffer_size = TRACE_SEQ_BUF_SIZE; + s->buffer = malloc_or_die(s->buffer_size); +} + +/** + * trace_seq_destroy - free up memory of a trace_seq + * @s: a pointer to the trace_seq to free the buffer + * + * Only frees the buffer, not the trace_seq struct itself. + */ +void trace_seq_destroy(struct trace_seq *s) +{ + if (!s) + return; + TRACE_SEQ_CHECK(s); + free(s->buffer); + s->buffer = TRACE_SEQ_POISON; +} + +static void expand_buffer(struct trace_seq *s) +{ + s->buffer_size += TRACE_SEQ_BUF_SIZE; + s->buffer = realloc(s->buffer, s->buffer_size); + if (!s->buffer) + die("Can't allocate trace_seq buffer memory"); +} + +/** + * trace_seq_printf - sequence printing of trace information + * @s: trace sequence descriptor + * @fmt: printf format string + * + * It returns 0 if the trace oversizes the buffer's free + * space, 1 otherwise. + * + * The tracer may use either sequence operations or its own + * copy to user routines. To simplify formating of a trace + * trace_seq_printf is used to store strings into a special + * buffer (@s). Then the output may be either used by + * the sequencer or pulled into another buffer. + */ +int +trace_seq_printf(struct trace_seq *s, const char *fmt, ...) +{ + va_list ap; + int len; + int ret; + + TRACE_SEQ_CHECK(s); + + try_again: + len = (s->buffer_size - 1) - s->len; + + va_start(ap, fmt); + ret = vsnprintf(s->buffer + s->len, len, fmt, ap); + va_end(ap); + + if (ret >= len) { + expand_buffer(s); + goto try_again; + } + + s->len += ret; + + return 1; +} + +/** + * trace_seq_vprintf - sequence printing of trace information + * @s: trace sequence descriptor + * @fmt: printf format string + * + * The tracer may use either sequence operations or its own + * copy to user routines. To simplify formating of a trace + * trace_seq_printf is used to store strings into a special + * buffer (@s). Then the output may be either used by + * the sequencer or pulled into another buffer. + */ +int +trace_seq_vprintf(struct trace_seq *s, const char *fmt, va_list args) +{ + int len; + int ret; + + TRACE_SEQ_CHECK(s); + + try_again: + len = (s->buffer_size - 1) - s->len; + + ret = vsnprintf(s->buffer + s->len, len, fmt, args); + + if (ret >= len) { + expand_buffer(s); + goto try_again; + } + + s->len += ret; + + return len; +} + +/** + * trace_seq_puts - trace sequence printing of simple string + * @s: trace sequence descriptor + * @str: simple string to record + * + * The tracer may use either the sequence operations or its own + * copy to user routines. This function records a simple string + * into a special buffer (@s) for later retrieval by a sequencer + * or other mechanism. + */ +int trace_seq_puts(struct trace_seq *s, const char *str) +{ + int len; + + TRACE_SEQ_CHECK(s); + + len = strlen(str); + + while (len > ((s->buffer_size - 1) - s->len)) + expand_buffer(s); + + memcpy(s->buffer + s->len, str, len); + s->len += len; + + return len; +} + +int trace_seq_putc(struct trace_seq *s, unsigned char c) +{ + TRACE_SEQ_CHECK(s); + + while (s->len >= (s->buffer_size - 1)) + expand_buffer(s); + + s->buffer[s->len++] = c; + + return 1; +} + +void trace_seq_terminate(struct trace_seq *s) +{ + TRACE_SEQ_CHECK(s); + + /* There's always one character left on the buffer */ + s->buffer[s->len] = 0; +} + +int trace_seq_do_printf(struct trace_seq *s) +{ + TRACE_SEQ_CHECK(s); + return printf("%.*s", s->len, s->buffer); +} diff --git a/tools/net/Makefile b/tools/net/Makefile new file mode 100644 index 000000000..b4444d53b --- /dev/null +++ b/tools/net/Makefile @@ -0,0 +1,15 @@ +prefix = /usr + +CC = gcc + +all : bpf_jit_disasm + +bpf_jit_disasm : CFLAGS = -Wall -O2 +bpf_jit_disasm : LDLIBS = -lopcodes -lbfd -ldl +bpf_jit_disasm : bpf_jit_disasm.o + +clean : + rm -rf *.o bpf_jit_disasm + +install : + install bpf_jit_disasm $(prefix)/bin/bpf_jit_disasm diff --git a/tools/net/bpf_jit_disasm.c b/tools/net/bpf_jit_disasm.c new file mode 100644 index 000000000..cfe0cdcda --- /dev/null +++ b/tools/net/bpf_jit_disasm.c @@ -0,0 +1,199 @@ +/* + * Minimal BPF JIT image disassembler + * + * Disassembles BPF JIT compiler emitted opcodes back to asm insn's for + * debugging or verification purposes. + * + * To get the disassembly of the JIT code, do the following: + * + * 1) `echo 2 > /proc/sys/net/core/bpf_jit_enable` + * 2) Load a BPF filter (e.g. `tcpdump -p -n -s 0 -i eth1 host 192.168.20.0/24`) + * 3) Run e.g. `bpf_jit_disasm -o` to read out the last JIT code + * + * Copyright 2013 Daniel Borkmann + * Licensed under the GNU General Public License, version 2.0 (GPLv2) + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +static void get_exec_path(char *tpath, size_t size) +{ + char *path; + ssize_t len; + + snprintf(tpath, size, "/proc/%d/exe", (int) getpid()); + tpath[size - 1] = 0; + + path = strdup(tpath); + assert(path); + + len = readlink(path, tpath, size); + tpath[len] = 0; + + free(path); +} + +static void get_asm_insns(uint8_t *image, size_t len, unsigned long base, + int opcodes) +{ + int count, i, pc = 0; + char tpath[256]; + struct disassemble_info info; + disassembler_ftype disassemble; + bfd *bfdf; + + memset(tpath, 0, sizeof(tpath)); + get_exec_path(tpath, sizeof(tpath)); + + bfdf = bfd_openr(tpath, NULL); + assert(bfdf); + assert(bfd_check_format(bfdf, bfd_object)); + + init_disassemble_info(&info, stdout, (fprintf_ftype) fprintf); + info.arch = bfd_get_arch(bfdf); + info.mach = bfd_get_mach(bfdf); + info.buffer = image; + info.buffer_length = len; + + disassemble_init_for_target(&info); + + disassemble = disassembler(bfdf); + assert(disassemble); + + do { + printf("%4x:\t", pc); + + count = disassemble(pc, &info); + + if (opcodes) { + printf("\n\t"); + for (i = 0; i < count; ++i) + printf("%02x ", (uint8_t) image[pc + i]); + } + printf("\n"); + + pc += count; + } while(count > 0 && pc < len); + + bfd_close(bfdf); +} + +static char *get_klog_buff(int *klen) +{ + int ret, len = klogctl(10, NULL, 0); + char *buff = malloc(len); + + assert(buff && klen); + ret = klogctl(3, buff, len); + assert(ret >= 0); + *klen = ret; + + return buff; +} + +static void put_klog_buff(char *buff) +{ + free(buff); +} + +static int get_last_jit_image(char *haystack, size_t hlen, + uint8_t *image, size_t ilen, + unsigned long *base) +{ + char *ptr, *pptr, *tmp; + off_t off = 0; + int ret, flen, proglen, pass, ulen = 0; + regmatch_t pmatch[1]; + regex_t regex; + + if (hlen == 0) + return 0; + + ret = regcomp(®ex, "flen=[[:alnum:]]+ proglen=[[:digit:]]+ " + "pass=[[:digit:]]+ image=[[:xdigit:]]+", REG_EXTENDED); + assert(ret == 0); + + ptr = haystack; + while (1) { + ret = regexec(®ex, ptr, 1, pmatch, 0); + if (ret == 0) { + ptr += pmatch[0].rm_eo; + off += pmatch[0].rm_eo; + assert(off < hlen); + } else + break; + } + + ptr = haystack + off - (pmatch[0].rm_eo - pmatch[0].rm_so); + ret = sscanf(ptr, "flen=%d proglen=%d pass=%d image=%lx", + &flen, &proglen, &pass, base); + if (ret != 4) + return 0; + + tmp = ptr = haystack + off; + while ((ptr = strtok(tmp, "\n")) != NULL && ulen < ilen) { + tmp = NULL; + if (!strstr(ptr, "JIT code")) + continue; + pptr = ptr; + while ((ptr = strstr(pptr, ":"))) + pptr = ptr + 1; + ptr = pptr; + do { + image[ulen++] = (uint8_t) strtoul(pptr, &pptr, 16); + if (ptr == pptr || ulen >= ilen) { + ulen--; + break; + } + ptr = pptr; + } while (1); + } + + assert(ulen == proglen); + printf("%d bytes emitted from JIT compiler (pass:%d, flen:%d)\n", + proglen, pass, flen); + printf("%lx + :\n", *base); + + regfree(®ex); + return ulen; +} + +int main(int argc, char **argv) +{ + int len, klen, opcodes = 0; + char *kbuff; + unsigned long base; + uint8_t image[4096]; + + if (argc > 1) { + if (!strncmp("-o", argv[argc - 1], 2)) { + opcodes = 1; + } else { + printf("usage: bpf_jit_disasm [-o: show opcodes]\n"); + exit(0); + } + } + + bfd_init(); + memset(image, 0, sizeof(image)); + + kbuff = get_klog_buff(&klen); + + len = get_last_jit_image(kbuff, klen, image, sizeof(image), &base); + if (len > 0 && base > 0) + get_asm_insns(image, len, base, opcodes); + + put_klog_buff(kbuff); + + return 0; +} diff --git a/tools/nfsd/inject_fault.sh b/tools/nfsd/inject_fault.sh new file mode 100755 index 000000000..06a399ac8 --- /dev/null +++ b/tools/nfsd/inject_fault.sh @@ -0,0 +1,49 @@ +#!/bin/bash +# +# Copyright (c) 2011 Bryan Schumaker +# +# Script for easier NFSD fault injection + +# Check that debugfs has been mounted +DEBUGFS=`cat /proc/mounts | grep debugfs` +if [ "$DEBUGFS" == "" ]; then + echo "debugfs does not appear to be mounted!" + echo "Please mount debugfs and try again" + exit 1 +fi + +# Check that the fault injection directory exists +DEBUGDIR=`echo $DEBUGFS | awk '{print $2}'`/nfsd +if [ ! -d "$DEBUGDIR" ]; then + echo "$DEBUGDIR does not exist" + echo "Check that your .config selects CONFIG_NFSD_FAULT_INJECTION" + exit 1 +fi + +function help() +{ + echo "Usage $0 injection_type [count]" + echo "" + echo "Injection types are:" + ls $DEBUGDIR + exit 1 +} + +if [ $# == 0 ]; then + help +elif [ ! -f $DEBUGDIR/$1 ]; then + help +elif [ $# != 2 ]; then + COUNT=0 +else + COUNT=$2 +fi + +BEFORE=`mktemp` +AFTER=`mktemp` +dmesg > $BEFORE +echo $COUNT > $DEBUGDIR/$1 +dmesg > $AFTER +# Capture lines that only exist in the $AFTER file +diff $BEFORE $AFTER | grep ">" +rm -f $BEFORE $AFTER diff --git a/tools/perf/.gitignore b/tools/perf/.gitignore new file mode 100644 index 000000000..8f8fbc227 --- /dev/null +++ b/tools/perf/.gitignore @@ -0,0 +1,25 @@ +PERF-CFLAGS +PERF-GUI-VARS +PERF-VERSION-FILE +perf +perf-help +perf-record +perf-report +perf-stat +perf-top +perf*.1 +perf*.xml +perf*.html +common-cmds.h +perf.data +perf.data.old +perf-archive +tags +TAGS +cscope* +config.mak +config.mak.autogen +*-bison.* +*-flex.* +*.pyc +*.pyo diff --git a/tools/perf/CREDITS b/tools/perf/CREDITS new file mode 100644 index 000000000..c2ddcb3ac --- /dev/null +++ b/tools/perf/CREDITS @@ -0,0 +1,30 @@ +Most of the infrastructure that 'perf' uses here has been reused +from the Git project, as of version: + + 66996ec: Sync with 1.6.2.4 + +Here is an (incomplete!) list of main contributors to those files +in util/* and elsewhere: + + Alex Riesen + Christian Couder + Dmitry Potapov + Jeff King + Johannes Schindelin + Johannes Sixt + Junio C Hamano + Linus Torvalds + Matthias Kestenholz + Michal Ostrowski + Miklos Vajna + Petr Baudis + Pierre Habouzit + René Scharfe + Samuel Tardieu + Shawn O. Pearce + Steffen Prohaska + Steve Haslam + +Thanks guys! + +The full history of the files can be found in the upstream Git commits. diff --git a/tools/perf/Documentation/Makefile b/tools/perf/Documentation/Makefile new file mode 100644 index 000000000..eb30044a9 --- /dev/null +++ b/tools/perf/Documentation/Makefile @@ -0,0 +1,352 @@ +include ../config/utilities.mak + +OUTPUT := ./ +ifeq ("$(origin O)", "command line") + ifneq ($(O),) + OUTPUT := $(O)/ + endif +endif + +MAN1_TXT= \ + $(filter-out $(addsuffix .txt, $(ARTICLES) $(SP_ARTICLES)), \ + $(wildcard perf-*.txt)) \ + perf.txt +MAN5_TXT= +MAN7_TXT= + +MAN_TXT = $(MAN1_TXT) $(MAN5_TXT) $(MAN7_TXT) +_MAN_XML=$(patsubst %.txt,%.xml,$(MAN_TXT)) +_MAN_HTML=$(patsubst %.txt,%.html,$(MAN_TXT)) + +MAN_XML=$(addprefix $(OUTPUT),$(_MAN_XML)) +MAN_HTML=$(addprefix $(OUTPUT),$(_MAN_HTML)) + +ARTICLES = +# with their own formatting rules. +SP_ARTICLES = +API_DOCS = $(patsubst %.txt,%,$(filter-out technical/api-index-skel.txt technical/api-index.txt, $(wildcard technical/api-*.txt))) +SP_ARTICLES += $(API_DOCS) +SP_ARTICLES += technical/api-index + +_DOC_HTML = $(_MAN_HTML) +_DOC_HTML+=$(patsubst %,%.html,$(ARTICLES) $(SP_ARTICLES)) +DOC_HTML=$(addprefix $(OUTPUT),$(_DOC_HTML)) + +_DOC_MAN1=$(patsubst %.txt,%.1,$(MAN1_TXT)) +_DOC_MAN5=$(patsubst %.txt,%.5,$(MAN5_TXT)) +_DOC_MAN7=$(patsubst %.txt,%.7,$(MAN7_TXT)) + +DOC_MAN1=$(addprefix $(OUTPUT),$(_DOC_MAN1)) +DOC_MAN5=$(addprefix $(OUTPUT),$(_DOC_MAN5)) +DOC_MAN7=$(addprefix $(OUTPUT),$(_DOC_MAN7)) + +# Make the path relative to DESTDIR, not prefix +ifndef DESTDIR +prefix?=$(HOME) +endif +bindir?=$(prefix)/bin +htmldir?=$(prefix)/share/doc/perf-doc +pdfdir?=$(prefix)/share/doc/perf-doc +mandir?=$(prefix)/share/man +man1dir=$(mandir)/man1 +man5dir=$(mandir)/man5 +man7dir=$(mandir)/man7 + +ASCIIDOC=asciidoc +ASCIIDOC_EXTRA = --unsafe +MANPAGE_XSL = manpage-normal.xsl +XMLTO_EXTRA = +INSTALL?=install +RM ?= rm -f +DOC_REF = origin/man +HTML_REF = origin/html + +infodir?=$(prefix)/share/info +MAKEINFO=makeinfo +INSTALL_INFO=install-info +DOCBOOK2X_TEXI=docbook2x-texi +DBLATEX=dblatex +XMLTO=xmlto +ifndef PERL_PATH + PERL_PATH = /usr/bin/perl +endif + +-include ../config.mak.autogen +-include ../config.mak + +_tmp_tool_path := $(call get-executable,$(ASCIIDOC)) +ifeq ($(_tmp_tool_path),) + missing_tools = $(ASCIIDOC) +endif + +_tmp_tool_path := $(call get-executable,$(XMLTO)) +ifeq ($(_tmp_tool_path),) + missing_tools += $(XMLTO) +endif + +# +# For asciidoc ... +# -7.1.2, no extra settings are needed. +# 8.0-, set ASCIIDOC8. +# + +# +# For docbook-xsl ... +# -1.68.1, set ASCIIDOC_NO_ROFF? (based on changelog from 1.73.0) +# 1.69.0, no extra settings are needed? +# 1.69.1-1.71.0, set DOCBOOK_SUPPRESS_SP? +# 1.71.1, no extra settings are needed? +# 1.72.0, set DOCBOOK_XSL_172. +# 1.73.0-, set ASCIIDOC_NO_ROFF +# + +# +# If you had been using DOCBOOK_XSL_172 in an attempt to get rid +# of 'the ".ft C" problem' in your generated manpages, and you +# instead ended up with weird characters around callouts, try +# using ASCIIDOC_NO_ROFF instead (it works fine with ASCIIDOC8). +# + +ifdef ASCIIDOC8 +ASCIIDOC_EXTRA += -a asciidoc7compatible +endif +ifdef DOCBOOK_XSL_172 +ASCIIDOC_EXTRA += -a perf-asciidoc-no-roff +MANPAGE_XSL = manpage-1.72.xsl +else + ifdef ASCIIDOC_NO_ROFF + # docbook-xsl after 1.72 needs the regular XSL, but will not + # pass-thru raw roff codes from asciidoc.conf, so turn them off. + ASCIIDOC_EXTRA += -a perf-asciidoc-no-roff + endif +endif +ifdef MAN_BOLD_LITERAL +XMLTO_EXTRA += -m manpage-bold-literal.xsl +endif +ifdef DOCBOOK_SUPPRESS_SP +XMLTO_EXTRA += -m manpage-suppress-sp.xsl +endif + +SHELL_PATH ?= $(SHELL) +# Shell quote; +SHELL_PATH_SQ = $(subst ','\'',$(SHELL_PATH)) + +# +# Please note that there is a minor bug in asciidoc. +# The version after 6.0.3 _will_ include the patch found here: +# http://marc.theaimsgroup.com/?l=perf&m=111558757202243&w=2 +# +# Until that version is released you may have to apply the patch +# yourself - yes, all 6 characters of it! +# + +QUIET_SUBDIR0 = +$(MAKE) -C # space to separate -C and subdir +QUIET_SUBDIR1 = + +ifneq ($(findstring $(MAKEFLAGS),w),w) +PRINT_DIR = --no-print-directory +else # "make -w" +NO_SUBDIR = : +endif + +ifneq ($(findstring $(MAKEFLAGS),s),s) +ifndef V + QUIET_ASCIIDOC = @echo ' ' ASCIIDOC $@; + QUIET_XMLTO = @echo ' ' XMLTO $@; + QUIET_DB2TEXI = @echo ' ' DB2TEXI $@; + QUIET_MAKEINFO = @echo ' ' MAKEINFO $@; + QUIET_DBLATEX = @echo ' ' DBLATEX $@; + QUIET_XSLTPROC = @echo ' ' XSLTPROC $@; + QUIET_GEN = @echo ' ' GEN $@; + QUIET_STDERR = 2> /dev/null + QUIET_SUBDIR0 = +@subdir= + QUIET_SUBDIR1 = ;$(NO_SUBDIR) echo ' ' SUBDIR $$subdir; \ + $(MAKE) $(PRINT_DIR) -C $$subdir + export V +endif +endif + +all: html man + +html: $(DOC_HTML) + +$(DOC_HTML) $(DOC_MAN1) $(DOC_MAN5) $(DOC_MAN7): asciidoc.conf + +man: man1 man5 man7 +man1: $(DOC_MAN1) +man5: $(DOC_MAN5) +man7: $(DOC_MAN7) + +info: $(OUTPUT)perf.info $(OUTPUT)perfman.info + +pdf: $(OUTPUT)user-manual.pdf + +install: install-man + +check-man-tools: +ifdef missing_tools + $(error "You need to install $(missing_tools) for man pages") +endif + +do-install-man: man + $(INSTALL) -d -m 755 $(DESTDIR)$(man1dir) +# $(INSTALL) -d -m 755 $(DESTDIR)$(man5dir) +# $(INSTALL) -d -m 755 $(DESTDIR)$(man7dir) + $(INSTALL) -m 644 $(DOC_MAN1) $(DESTDIR)$(man1dir) +# $(INSTALL) -m 644 $(DOC_MAN5) $(DESTDIR)$(man5dir) +# $(INSTALL) -m 644 $(DOC_MAN7) $(DESTDIR)$(man7dir) + +install-man: check-man-tools man + +try-install-man: +ifdef missing_tools + $(warning Please install $(missing_tools) to have the man pages installed) +else + $(MAKE) do-install-man +endif + +install-info: info + $(INSTALL) -d -m 755 $(DESTDIR)$(infodir) + $(INSTALL) -m 644 $(OUTPUT)perf.info $(OUTPUT)perfman.info $(DESTDIR)$(infodir) + if test -r $(DESTDIR)$(infodir)/dir; then \ + $(INSTALL_INFO) --info-dir=$(DESTDIR)$(infodir) perf.info ;\ + $(INSTALL_INFO) --info-dir=$(DESTDIR)$(infodir) perfman.info ;\ + else \ + echo "No directory found in $(DESTDIR)$(infodir)" >&2 ; \ + fi + +install-pdf: pdf + $(INSTALL) -d -m 755 $(DESTDIR)$(pdfdir) + $(INSTALL) -m 644 $(OUTPUT)user-manual.pdf $(DESTDIR)$(pdfdir) + +#install-html: html +# '$(SHELL_PATH_SQ)' ./install-webdoc.sh $(DESTDIR)$(htmldir) + +ifneq ($(MAKECMDGOALS),clean) +ifneq ($(MAKECMDGOALS),tags) +$(OUTPUT)PERF-VERSION-FILE: .FORCE-PERF-VERSION-FILE + $(QUIET_SUBDIR0)../ $(QUIET_SUBDIR1) $(OUTPUT)PERF-VERSION-FILE + +-include $(OUTPUT)PERF-VERSION-FILE +endif +endif + +# +# Determine "include::" file references in asciidoc files. +# +$(OUTPUT)doc.dep : $(wildcard *.txt) build-docdep.perl + $(QUIET_GEN)$(RM) $@+ $@ && \ + $(PERL_PATH) ./build-docdep.perl >$@+ $(QUIET_STDERR) && \ + mv $@+ $@ + +-include $(OUPTUT)doc.dep + +_cmds_txt = cmds-ancillaryinterrogators.txt \ + cmds-ancillarymanipulators.txt \ + cmds-mainporcelain.txt \ + cmds-plumbinginterrogators.txt \ + cmds-plumbingmanipulators.txt \ + cmds-synchingrepositories.txt \ + cmds-synchelpers.txt \ + cmds-purehelpers.txt \ + cmds-foreignscminterface.txt +cmds_txt=$(addprefix $(OUTPUT),$(_cmds_txt)) + +$(cmds_txt): $(OUTPUT)cmd-list.made + +$(OUTPUT)cmd-list.made: cmd-list.perl ../command-list.txt $(MAN1_TXT) + $(QUIET_GEN)$(RM) $@ && \ + $(PERL_PATH) ./cmd-list.perl ../command-list.txt $(QUIET_STDERR) && \ + date >$@ + +clean: + $(RM) $(MAN_XML) $(addsuffix +,$(MAN_XML)) + $(RM) $(MAN_HTML) $(addsuffix +,$(MAN_HTML)) + $(RM) $(DOC_HTML) $(DOC_MAN1) $(DOC_MAN5) $(DOC_MAN7) + $(RM) $(OUTPUT)*.texi $(OUTPUT)*.texi+ $(OUTPUT)*.texi++ + $(RM) $(OUTPUT)perf.info $(OUTPUT)perfman.info + $(RM) $(OUTPUT)howto-index.txt $(OUTPUT)howto/*.html $(OUTPUT)doc.dep + $(RM) $(OUTPUT)technical/api-*.html $(OUTPUT)technical/api-index.txt + $(RM) $(cmds_txt) $(OUTPUT)*.made + +$(MAN_HTML): $(OUTPUT)%.html : %.txt + $(QUIET_ASCIIDOC)$(RM) $@+ $@ && \ + $(ASCIIDOC) -b xhtml11 -d manpage -f asciidoc.conf \ + $(ASCIIDOC_EXTRA) -aperf_version=$(PERF_VERSION) -o $@+ $< && \ + mv $@+ $@ + +$(OUTPUT)%.1 $(OUTPUT)%.5 $(OUTPUT)%.7 : $(OUTPUT)%.xml + $(QUIET_XMLTO)$(RM) $@ && \ + $(XMLTO) -o $(OUTPUT) -m $(MANPAGE_XSL) $(XMLTO_EXTRA) man $< + +$(OUTPUT)%.xml : %.txt + $(QUIET_ASCIIDOC)$(RM) $@+ $@ && \ + $(ASCIIDOC) -b docbook -d manpage -f asciidoc.conf \ + $(ASCIIDOC_EXTRA) -aperf_version=$(PERF_VERSION) -o $@+ $< && \ + mv $@+ $@ + +XSLT = docbook.xsl +XSLTOPTS = --xinclude --stringparam html.stylesheet docbook-xsl.css + +$(OUTPUT)user-manual.html: $(OUTPUT)user-manual.xml + $(QUIET_XSLTPROC)xsltproc $(XSLTOPTS) -o $@ $(XSLT) $< + +$(OUTPUT)perf.info: $(OUTPUT)user-manual.texi + $(QUIET_MAKEINFO)$(MAKEINFO) --no-split -o $@ $(OUTPUT)user-manual.texi + +$(OUTPUT)user-manual.texi: $(OUTPUT)user-manual.xml + $(QUIET_DB2TEXI)$(RM) $@+ $@ && \ + $(DOCBOOK2X_TEXI) $(OUTPUT)user-manual.xml --encoding=UTF-8 --to-stdout >$@++ && \ + $(PERL_PATH) fix-texi.perl <$@++ >$@+ && \ + rm $@++ && \ + mv $@+ $@ + +$(OUTPUT)user-manual.pdf: $(OUTPUT)user-manual.xml + $(QUIET_DBLATEX)$(RM) $@+ $@ && \ + $(DBLATEX) -o $@+ -p /etc/asciidoc/dblatex/asciidoc-dblatex.xsl -s /etc/asciidoc/dblatex/asciidoc-dblatex.sty $< && \ + mv $@+ $@ + +$(OUTPUT)perfman.texi: $(MAN_XML) cat-texi.perl + $(QUIET_DB2TEXI)$(RM) $@+ $@ && \ + ($(foreach xml,$(MAN_XML),$(DOCBOOK2X_TEXI) --encoding=UTF-8 \ + --to-stdout $(xml) &&) true) > $@++ && \ + $(PERL_PATH) cat-texi.perl $@ <$@++ >$@+ && \ + rm $@++ && \ + mv $@+ $@ + +$(OUTPUT)perfman.info: $(OUTPUT)perfman.texi + $(QUIET_MAKEINFO)$(MAKEINFO) --no-split --no-validate $*.texi + +$(patsubst %.txt,%.texi,$(MAN_TXT)): %.texi : %.xml + $(QUIET_DB2TEXI)$(RM) $@+ $@ && \ + $(DOCBOOK2X_TEXI) --to-stdout $*.xml >$@+ && \ + mv $@+ $@ + +howto-index.txt: howto-index.sh $(wildcard howto/*.txt) + $(QUIET_GEN)$(RM) $@+ $@ && \ + '$(SHELL_PATH_SQ)' ./howto-index.sh $(wildcard howto/*.txt) >$@+ && \ + mv $@+ $@ + +$(patsubst %,%.html,$(ARTICLES)) : %.html : %.txt + $(QUIET_ASCIIDOC)$(ASCIIDOC) -b xhtml11 $*.txt + +WEBDOC_DEST = /pub/software/tools/perf/docs + +$(patsubst %.txt,%.html,$(wildcard howto/*.txt)): %.html : %.txt + $(QUIET_ASCIIDOC)$(RM) $@+ $@ && \ + sed -e '1,/^$$/d' $< | $(ASCIIDOC) -b xhtml11 - >$@+ && \ + mv $@+ $@ + +# UNIMPLEMENTED +#install-webdoc : html +# '$(SHELL_PATH_SQ)' ./install-webdoc.sh $(WEBDOC_DEST) + +# quick-install: quick-install-man + +# quick-install-man: +# '$(SHELL_PATH_SQ)' ./install-doc-quick.sh $(DOC_REF) $(DESTDIR)$(mandir) + +#quick-install-html: +# '$(SHELL_PATH_SQ)' ./install-doc-quick.sh $(HTML_REF) $(DESTDIR)$(htmldir) + +.PHONY: .FORCE-PERF-VERSION-FILE diff --git a/tools/perf/Documentation/android.txt b/tools/perf/Documentation/android.txt new file mode 100644 index 000000000..8484c3a04 --- /dev/null +++ b/tools/perf/Documentation/android.txt @@ -0,0 +1,78 @@ +How to compile perf for Android +========================================= + +I. Set the Android NDK environment +------------------------------------------------ + +(a). Use the Android NDK +------------------------------------------------ +1. You need to download and install the Android Native Development Kit (NDK). +Set the NDK variable to point to the path where you installed the NDK: + export NDK=/path/to/android-ndk + +2. Set cross-compiling environment variables for NDK toolchain and sysroot. +For arm: + export NDK_TOOLCHAIN=${NDK}/toolchains/arm-linux-androideabi-4.6/prebuilt/linux-x86/bin/arm-linux-androideabi- + export NDK_SYSROOT=${NDK}/platforms/android-9/arch-arm +For x86: + export NDK_TOOLCHAIN=${NDK}/toolchains/x86-4.6/prebuilt/linux-x86/bin/i686-linux-android- + export NDK_SYSROOT=${NDK}/platforms/android-9/arch-x86 + +This method is not working for Android NDK versions up to Revision 8b. +perf uses some bionic enhancements that are not included in these NDK versions. +You can use method (b) described below instead. + +(b). Use the Android source tree +----------------------------------------------- +1. Download the master branch of the Android source tree. +Set the environment for the target you want using: + source build/envsetup.sh + lunch + +2. Build your own NDK sysroot to contain latest bionic changes and set the +NDK sysroot environment variable. + cd ${ANDROID_BUILD_TOP}/ndk +For arm: + ./build/tools/build-ndk-sysroot.sh --abi=arm + export NDK_SYSROOT=${ANDROID_BUILD_TOP}/ndk/build/platforms/android-3/arch-arm +For x86: + ./build/tools/build-ndk-sysroot.sh --abi=x86 + export NDK_SYSROOT=${ANDROID_BUILD_TOP}/ndk/build/platforms/android-3/arch-x86 + +3. Set the NDK toolchain environment variable. +For arm: + export NDK_TOOLCHAIN=${ANDROID_TOOLCHAIN}/arm-linux-androideabi- +For x86: + export NDK_TOOLCHAIN=${ANDROID_TOOLCHAIN}/i686-linux-android- + +II. Compile perf for Android +------------------------------------------------ +You need to run make with the NDK toolchain and sysroot defined above: +For arm: + make ARCH=arm CROSS_COMPILE=${NDK_TOOLCHAIN} CFLAGS="--sysroot=${NDK_SYSROOT}" +For x86: + make ARCH=x86 CROSS_COMPILE=${NDK_TOOLCHAIN} CFLAGS="--sysroot=${NDK_SYSROOT}" + +III. Install perf +----------------------------------------------- +You need to connect to your Android device/emulator using adb. +Install perf using: + adb push perf /data/perf + +If you also want to use perf-archive you need busybox tools for Android. +For installing perf-archive, you first need to replace #!/bin/bash with #!/system/bin/sh: + sed 's/#!\/bin\/bash/#!\/system\/bin\/sh/g' perf-archive >> /tmp/perf-archive + chmod +x /tmp/perf-archive + adb push /tmp/perf-archive /data/perf-archive + +IV. Environment settings for running perf +------------------------------------------------ +Some perf features need environment variables to run properly. +You need to set these before running perf on the target: + adb shell + # PERF_PAGER=cat + +IV. Run perf +------------------------------------------------ +Run perf on your device/emulator to which you previously connected using adb: + # ./data/perf diff --git a/tools/perf/Documentation/asciidoc.conf b/tools/perf/Documentation/asciidoc.conf new file mode 100644 index 000000000..356b23a40 --- /dev/null +++ b/tools/perf/Documentation/asciidoc.conf @@ -0,0 +1,91 @@ +## linkperf: macro +# +# Usage: linkperf:command[manpage-section] +# +# Note, {0} is the manpage section, while {target} is the command. +# +# Show PERF link as: (
); if section is defined, else just show +# the command. + +[macros] +(?su)[\\]?(?Plinkperf):(?P\S*?)\[(?P.*?)\]= + +[attributes] +asterisk=* +plus=+ +caret=^ +startsb=[ +endsb=] +tilde=~ + +ifdef::backend-docbook[] +[linkperf-inlinemacro] +{0%{target}} +{0#} +{0#{target}{0}} +{0#} +endif::backend-docbook[] + +ifdef::backend-docbook[] +ifndef::perf-asciidoc-no-roff[] +# "unbreak" docbook-xsl v1.68 for manpages. v1.69 works with or without this. +# v1.72 breaks with this because it replaces dots not in roff requests. +[listingblock] +{title} + +ifdef::doctype-manpage[] + .ft C +endif::doctype-manpage[] +| +ifdef::doctype-manpage[] + .ft +endif::doctype-manpage[] + +{title#} +endif::perf-asciidoc-no-roff[] + +ifdef::perf-asciidoc-no-roff[] +ifdef::doctype-manpage[] +# The following two small workarounds insert a simple paragraph after screen +[listingblock] +{title} + +| + +{title#} + +[verseblock] +{title} +{title%} +{title#} +| + +{title#} +{title%} +endif::doctype-manpage[] +endif::perf-asciidoc-no-roff[] +endif::backend-docbook[] + +ifdef::doctype-manpage[] +ifdef::backend-docbook[] +[header] +template::[header-declarations] + + +{mantitle} +{manvolnum} +perf +{perf_version} +perf Manual + + + {manname} + {manpurpose} + +endif::backend-docbook[] +endif::doctype-manpage[] + +ifdef::backend-xhtml11[] +[linkperf-inlinemacro] +{target}{0?({0})} +endif::backend-xhtml11[] diff --git a/tools/perf/Documentation/examples.txt b/tools/perf/Documentation/examples.txt new file mode 100644 index 000000000..77f952762 --- /dev/null +++ b/tools/perf/Documentation/examples.txt @@ -0,0 +1,225 @@ + + ------------------------------ + ****** perf by examples ****** + ------------------------------ + +[ From an e-mail by Ingo Molnar, http://lkml.org/lkml/2009/8/4/346 ] + + +First, discovery/enumeration of available counters can be done via +'perf list': + +titan:~> perf list + [...] + kmem:kmalloc [Tracepoint event] + kmem:kmem_cache_alloc [Tracepoint event] + kmem:kmalloc_node [Tracepoint event] + kmem:kmem_cache_alloc_node [Tracepoint event] + kmem:kfree [Tracepoint event] + kmem:kmem_cache_free [Tracepoint event] + kmem:mm_page_free [Tracepoint event] + kmem:mm_page_free_batched [Tracepoint event] + kmem:mm_page_alloc [Tracepoint event] + kmem:mm_page_alloc_zone_locked [Tracepoint event] + kmem:mm_page_pcpu_drain [Tracepoint event] + kmem:mm_page_alloc_extfrag [Tracepoint event] + +Then any (or all) of the above event sources can be activated and +measured. For example the page alloc/free properties of a 'hackbench +run' are: + + titan:~> perf stat -e kmem:mm_page_pcpu_drain -e kmem:mm_page_alloc + -e kmem:mm_page_free_batched -e kmem:mm_page_free ./hackbench 10 + Time: 0.575 + + Performance counter stats for './hackbench 10': + + 13857 kmem:mm_page_pcpu_drain + 27576 kmem:mm_page_alloc + 6025 kmem:mm_page_free_batched + 20934 kmem:mm_page_free + + 0.613972165 seconds time elapsed + +You can observe the statistical properties as well, by using the +'repeat the workload N times' feature of perf stat: + + titan:~> perf stat --repeat 5 -e kmem:mm_page_pcpu_drain -e + kmem:mm_page_alloc -e kmem:mm_page_free_batched -e + kmem:mm_page_free ./hackbench 10 + Time: 0.627 + Time: 0.644 + Time: 0.564 + Time: 0.559 + Time: 0.626 + + Performance counter stats for './hackbench 10' (5 runs): + + 12920 kmem:mm_page_pcpu_drain ( +- 3.359% ) + 25035 kmem:mm_page_alloc ( +- 3.783% ) + 6104 kmem:mm_page_free_batched ( +- 0.934% ) + 18376 kmem:mm_page_free ( +- 4.941% ) + + 0.643954516 seconds time elapsed ( +- 2.363% ) + +Furthermore, these tracepoints can be used to sample the workload as +well. For example the page allocations done by a 'git gc' can be +captured the following way: + + titan:~/git> perf record -f -e kmem:mm_page_alloc -c 1 ./git gc + Counting objects: 1148, done. + Delta compression using up to 2 threads. + Compressing objects: 100% (450/450), done. + Writing objects: 100% (1148/1148), done. + Total 1148 (delta 690), reused 1148 (delta 690) + [ perf record: Captured and wrote 0.267 MB perf.data (~11679 samples) ] + +To check which functions generated page allocations: + + titan:~/git> perf report + # Samples: 10646 + # + # Overhead Command Shared Object + # ........ ............... .......................... + # + 23.57% git-repack /lib64/libc-2.5.so + 21.81% git /lib64/libc-2.5.so + 14.59% git ./git + 11.79% git-repack ./git + 7.12% git /lib64/ld-2.5.so + 3.16% git-repack /lib64/libpthread-2.5.so + 2.09% git-repack /bin/bash + 1.97% rm /lib64/libc-2.5.so + 1.39% mv /lib64/ld-2.5.so + 1.37% mv /lib64/libc-2.5.so + 1.12% git-repack /lib64/ld-2.5.so + 0.95% rm /lib64/ld-2.5.so + 0.90% git-update-serv /lib64/libc-2.5.so + 0.73% git-update-serv /lib64/ld-2.5.so + 0.68% perf /lib64/libpthread-2.5.so + 0.64% git-repack /usr/lib64/libz.so.1.2.3 + +Or to see it on a more finegrained level: + +titan:~/git> perf report --sort comm,dso,symbol +# Samples: 10646 +# +# Overhead Command Shared Object Symbol +# ........ ............... .......................... ...... +# + 9.35% git-repack ./git [.] insert_obj_hash + 9.12% git ./git [.] insert_obj_hash + 7.31% git /lib64/libc-2.5.so [.] memcpy + 6.34% git-repack /lib64/libc-2.5.so [.] _int_malloc + 6.24% git-repack /lib64/libc-2.5.so [.] memcpy + 5.82% git-repack /lib64/libc-2.5.so [.] __GI___fork + 5.47% git /lib64/libc-2.5.so [.] _int_malloc + 2.99% git /lib64/libc-2.5.so [.] memset + +Furthermore, call-graph sampling can be done too, of page +allocations - to see precisely what kind of page allocations there +are: + + titan:~/git> perf record -f -g -e kmem:mm_page_alloc -c 1 ./git gc + Counting objects: 1148, done. + Delta compression using up to 2 threads. + Compressing objects: 100% (450/450), done. + Writing objects: 100% (1148/1148), done. + Total 1148 (delta 690), reused 1148 (delta 690) + [ perf record: Captured and wrote 0.963 MB perf.data (~42069 samples) ] + + titan:~/git> perf report -g + # Samples: 10686 + # + # Overhead Command Shared Object + # ........ ............... .......................... + # + 23.25% git-repack /lib64/libc-2.5.so + | + |--50.00%-- _int_free + | + |--37.50%-- __GI___fork + | make_child + | + |--12.50%-- ptmalloc_unlock_all2 + | make_child + | + --6.25%-- __GI_strcpy + 21.61% git /lib64/libc-2.5.so + | + |--30.00%-- __GI_read + | | + | --83.33%-- git_config_from_file + | git_config + | | + [...] + +Or you can observe the whole system's page allocations for 10 +seconds: + +titan:~/git> perf stat -a -e kmem:mm_page_pcpu_drain -e +kmem:mm_page_alloc -e kmem:mm_page_free_batched -e +kmem:mm_page_free sleep 10 + + Performance counter stats for 'sleep 10': + + 171585 kmem:mm_page_pcpu_drain + 322114 kmem:mm_page_alloc + 73623 kmem:mm_page_free_batched + 254115 kmem:mm_page_free + + 10.000591410 seconds time elapsed + +Or observe how fluctuating the page allocations are, via statistical +analysis done over ten 1-second intervals: + + titan:~/git> perf stat --repeat 10 -a -e kmem:mm_page_pcpu_drain -e + kmem:mm_page_alloc -e kmem:mm_page_free_batched -e + kmem:mm_page_free sleep 1 + + Performance counter stats for 'sleep 1' (10 runs): + + 17254 kmem:mm_page_pcpu_drain ( +- 3.709% ) + 34394 kmem:mm_page_alloc ( +- 4.617% ) + 7509 kmem:mm_page_free_batched ( +- 4.820% ) + 25653 kmem:mm_page_free ( +- 3.672% ) + + 1.058135029 seconds time elapsed ( +- 3.089% ) + +Or you can annotate the recorded 'git gc' run on a per symbol basis +and check which instructions/source-code generated page allocations: + + titan:~/git> perf annotate __GI___fork + ------------------------------------------------ + Percent | Source code & Disassembly of libc-2.5.so + ------------------------------------------------ + : + : + : Disassembly of section .plt: + : Disassembly of section .text: + : + : 00000031a2e95560 <__fork>: + [...] + 0.00 : 31a2e95602: b8 38 00 00 00 mov $0x38,%eax + 0.00 : 31a2e95607: 0f 05 syscall + 83.42 : 31a2e95609: 48 3d 00 f0 ff ff cmp $0xfffffffffffff000,%rax + 0.00 : 31a2e9560f: 0f 87 4d 01 00 00 ja 31a2e95762 <__fork+0x202> + 0.00 : 31a2e95615: 85 c0 test %eax,%eax + +( this shows that 83.42% of __GI___fork's page allocations come from + the 0x38 system call it performs. ) + +etc. etc. - a lot more is possible. I could list a dozen of +other different usecases straight away - neither of which is +possible via /proc/vmstat. + +/proc/vmstat is not in the same league really, in terms of +expressive power of system analysis and performance +analysis. + +All that the above results needed were those new tracepoints +in include/tracing/events/kmem.h. + + Ingo + + diff --git a/tools/perf/Documentation/jit-interface.txt b/tools/perf/Documentation/jit-interface.txt new file mode 100644 index 000000000..a8656f564 --- /dev/null +++ b/tools/perf/Documentation/jit-interface.txt @@ -0,0 +1,15 @@ +perf supports a simple JIT interface to resolve symbols for dynamic code generated +by a JIT. + +The JIT has to write a /tmp/perf-%d.map (%d = pid of process) file + +This is a text file. + +Each line has the following format, fields separated with spaces: + +START SIZE symbolname + +START and SIZE are hex numbers without 0x. +symbolname is the rest of the line, so it could contain special characters. + +The ownership of the file has to match the process. diff --git a/tools/perf/Documentation/manpage-1.72.xsl b/tools/perf/Documentation/manpage-1.72.xsl new file mode 100644 index 000000000..b4d315cb8 --- /dev/null +++ b/tools/perf/Documentation/manpage-1.72.xsl @@ -0,0 +1,14 @@ + + + + + + + + + + diff --git a/tools/perf/Documentation/manpage-base.xsl b/tools/perf/Documentation/manpage-base.xsl new file mode 100644 index 000000000..a264fa616 --- /dev/null +++ b/tools/perf/Documentation/manpage-base.xsl @@ -0,0 +1,35 @@ + + + + + + + + + + + + + + sp + + + + + + + + br + + + diff --git a/tools/perf/Documentation/manpage-bold-literal.xsl b/tools/perf/Documentation/manpage-bold-literal.xsl new file mode 100644 index 000000000..608eb5df6 --- /dev/null +++ b/tools/perf/Documentation/manpage-bold-literal.xsl @@ -0,0 +1,17 @@ + + + + + + + fB + + + fR + + + diff --git a/tools/perf/Documentation/manpage-normal.xsl b/tools/perf/Documentation/manpage-normal.xsl new file mode 100644 index 000000000..a48f5b11f --- /dev/null +++ b/tools/perf/Documentation/manpage-normal.xsl @@ -0,0 +1,13 @@ + + + + + + +\ +. + + diff --git a/tools/perf/Documentation/manpage-suppress-sp.xsl b/tools/perf/Documentation/manpage-suppress-sp.xsl new file mode 100644 index 000000000..a63c7632a --- /dev/null +++ b/tools/perf/Documentation/manpage-suppress-sp.xsl @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + diff --git a/tools/perf/Documentation/perf-annotate.txt b/tools/perf/Documentation/perf-annotate.txt new file mode 100644 index 000000000..e9cd39a92 --- /dev/null +++ b/tools/perf/Documentation/perf-annotate.txt @@ -0,0 +1,101 @@ +perf-annotate(1) +================ + +NAME +---- +perf-annotate - Read perf.data (created by perf record) and display annotated code + +SYNOPSIS +-------- +[verse] +'perf annotate' [-i | --input=file] [symbol_name] + +DESCRIPTION +----------- +This command reads the input file and displays an annotated version of the +code. If the object file has debug symbols then the source code will be +displayed alongside assembly code. + +If there is no debug info in the object, then annotated assembly is displayed. + +OPTIONS +------- +-i:: +--input=:: + Input file name. (default: perf.data unless stdin is a fifo) + +-d:: +--dsos=:: + Only consider symbols in these dsos. +-s:: +--symbol=:: + Symbol to annotate. + +-f:: +--force:: + Don't complain, do it. + +-v:: +--verbose:: + Be more verbose. (Show symbol address, etc) + +-D:: +--dump-raw-trace:: + Dump raw trace in ASCII. + +-k:: +--vmlinux=:: + vmlinux pathname. + +-m:: +--modules:: + Load module symbols. WARNING: use only with -k and LIVE kernel. + +-l:: +--print-line:: + Print matching source lines (may be slow). + +-P:: +--full-paths:: + Don't shorten the displayed pathnames. + +--stdio:: Use the stdio interface. + +--tui:: Use the TUI interface. Use of --tui requires a tty, if one is not + present, as when piping to other commands, the stdio interface is + used. This interfaces starts by centering on the line with more + samples, TAB/UNTAB cycles through the lines with more samples. + +--gtk:: Use the GTK interface. + +-C:: +--cpu:: Only report samples for the list of CPUs provided. Multiple CPUs can + be provided as a comma-separated list with no space: 0,1. Ranges of + CPUs are specified with -: 0-2. Default is to report samples on all + CPUs. + +--asm-raw:: + Show raw instruction encoding of assembly instructions. + +--source:: + Interleave source code with assembly code. Enabled by default, + disable with --no-source. + +--symfs=:: + Look for files with symbols relative to this directory. + +-M:: +--disassembler-style=:: Set disassembler style for objdump. + +--objdump=:: + Path to objdump binary. + +--skip-missing:: + Skip symbols that cannot be annotated. + +--group:: + Show event group information together + +SEE ALSO +-------- +linkperf:perf-record[1], linkperf:perf-report[1] diff --git a/tools/perf/Documentation/perf-archive.txt b/tools/perf/Documentation/perf-archive.txt new file mode 100644 index 000000000..fae174dc7 --- /dev/null +++ b/tools/perf/Documentation/perf-archive.txt @@ -0,0 +1,22 @@ +perf-archive(1) +=============== + +NAME +---- +perf-archive - Create archive with object files with build-ids found in perf.data file + +SYNOPSIS +-------- +[verse] +'perf archive' [file] + +DESCRIPTION +----------- +This command runs runs perf-buildid-list --with-hits, and collects the files +with the buildids found so that analisys of perf.data contents can be possible +on another machine. + + +SEE ALSO +-------- +linkperf:perf-record[1], linkperf:perf-buildid-list[1], linkperf:perf-report[1] diff --git a/tools/perf/Documentation/perf-bench.txt b/tools/perf/Documentation/perf-bench.txt new file mode 100644 index 000000000..7065cd6fb --- /dev/null +++ b/tools/perf/Documentation/perf-bench.txt @@ -0,0 +1,192 @@ +perf-bench(1) +============= + +NAME +---- +perf-bench - General framework for benchmark suites + +SYNOPSIS +-------- +[verse] +'perf bench' [] [] + +DESCRIPTION +----------- +This 'perf bench' command is a general framework for benchmark suites. + +COMMON OPTIONS +-------------- +-f:: +--format=:: +Specify format style. +Current available format styles are: + +'default':: +Default style. This is mainly for human reading. +--------------------- +% perf bench sched pipe # with no style specified +(executing 1000000 pipe operations between two tasks) + Total time:5.855 sec + 5.855061 usecs/op + 170792 ops/sec +--------------------- + +'simple':: +This simple style is friendly for automated +processing by scripts. +--------------------- +% perf bench --format=simple sched pipe # specified simple +5.988 +--------------------- + +SUBSYSTEM +--------- + +'sched':: + Scheduler and IPC mechanisms. + +'mem':: + Memory access performance. + +'all':: + All benchmark subsystems. + +SUITES FOR 'sched' +~~~~~~~~~~~~~~~~~~ +*messaging*:: +Suite for evaluating performance of scheduler and IPC mechanisms. +Based on hackbench by Rusty Russell. + +Options of *messaging* +^^^^^^^^^^^^^^^^^^^^^^ +-p:: +--pipe:: +Use pipe() instead of socketpair() + +-t:: +--thread:: +Be multi thread instead of multi process + +-g:: +--group=:: +Specify number of groups + +-l:: +--loop=:: +Specify number of loops + +Example of *messaging* +^^^^^^^^^^^^^^^^^^^^^^ + +--------------------- +% perf bench sched messaging # run with default +options (20 sender and receiver processes per group) +(10 groups == 400 processes run) + + Total time:0.308 sec + +% perf bench sched messaging -t -g 20 # be multi-thread, with 20 groups +(20 sender and receiver threads per group) +(20 groups == 800 threads run) + + Total time:0.582 sec +--------------------- + +*pipe*:: +Suite for pipe() system call. +Based on pipe-test-1m.c by Ingo Molnar. + +Options of *pipe* +^^^^^^^^^^^^^^^^^ +-l:: +--loop=:: +Specify number of loops. + +Example of *pipe* +^^^^^^^^^^^^^^^^^ + +--------------------- +% perf bench sched pipe +(executing 1000000 pipe operations between two tasks) + + Total time:8.091 sec + 8.091833 usecs/op + 123581 ops/sec + +% perf bench sched pipe -l 1000 # loop 1000 +(executing 1000 pipe operations between two tasks) + + Total time:0.016 sec + 16.948000 usecs/op + 59004 ops/sec +--------------------- + +SUITES FOR 'mem' +~~~~~~~~~~~~~~~~ +*memcpy*:: +Suite for evaluating performance of simple memory copy in various ways. + +Options of *memcpy* +^^^^^^^^^^^^^^^^^^^ +-l:: +--length:: +Specify length of memory to copy (default: 1MB). +Available units are B, KB, MB, GB and TB (case insensitive). + +-r:: +--routine:: +Specify routine to copy (default: default). +Available routines are depend on the architecture. +On x86-64, x86-64-unrolled, x86-64-movsq and x86-64-movsb are supported. + +-i:: +--iterations:: +Repeat memcpy invocation this number of times. + +-c:: +--cycle:: +Use perf's cpu-cycles event instead of gettimeofday syscall. + +-o:: +--only-prefault:: +Show only the result with page faults before memcpy. + +-n:: +--no-prefault:: +Show only the result without page faults before memcpy. + +*memset*:: +Suite for evaluating performance of simple memory set in various ways. + +Options of *memset* +^^^^^^^^^^^^^^^^^^^ +-l:: +--length:: +Specify length of memory to set (default: 1MB). +Available units are B, KB, MB, GB and TB (case insensitive). + +-r:: +--routine:: +Specify routine to set (default: default). +Available routines are depend on the architecture. +On x86-64, x86-64-unrolled, x86-64-stosq and x86-64-stosb are supported. + +-i:: +--iterations:: +Repeat memset invocation this number of times. + +-c:: +--cycle:: +Use perf's cpu-cycles event instead of gettimeofday syscall. + +-o:: +--only-prefault:: +Show only the result with page faults before memset. + +-n:: +--no-prefault:: +Show only the result without page faults before memset. + +SEE ALSO +-------- +linkperf:perf[1] diff --git a/tools/perf/Documentation/perf-buildid-cache.txt b/tools/perf/Documentation/perf-buildid-cache.txt new file mode 100644 index 000000000..e9a8349a7 --- /dev/null +++ b/tools/perf/Documentation/perf-buildid-cache.txt @@ -0,0 +1,40 @@ +perf-buildid-cache(1) +===================== + +NAME +---- +perf-buildid-cache - Manage build-id cache. + +SYNOPSIS +-------- +[verse] +'perf buildid-cache ' + +DESCRIPTION +----------- +This command manages the build-id cache. It can add and remove files to/from +the cache. In the future it should as well purge older entries, set upper +limits for the space used by the cache, etc. + +OPTIONS +------- +-a:: +--add=:: + Add specified file to the cache. +-r:: +--remove=:: + Remove specified file from the cache. +-M:: +--missing=:: + List missing build ids in the cache for the specified file. +-u:: +--update:: + Update specified file of the cache. It can be used to update kallsyms + kernel dso to vmlinux in order to support annotation. +-v:: +--verbose:: + Be more verbose. + +SEE ALSO +-------- +linkperf:perf-record[1], linkperf:perf-report[1], linkperf:perf-buildid-list[1] diff --git a/tools/perf/Documentation/perf-buildid-list.txt b/tools/perf/Documentation/perf-buildid-list.txt new file mode 100644 index 000000000..25c52efcc --- /dev/null +++ b/tools/perf/Documentation/perf-buildid-list.txt @@ -0,0 +1,43 @@ +perf-buildid-list(1) +==================== + +NAME +---- +perf-buildid-list - List the buildids in a perf.data file + +SYNOPSIS +-------- +[verse] +'perf buildid-list ' + +DESCRIPTION +----------- +This command displays the buildids found in a perf.data file, so that other +tools can be used to fetch packages with matching symbol tables for use by +perf report. + +It can also be used to show the build id of the running kernel or in an ELF +file using -i/--input. + +OPTIONS +------- +-H:: +--with-hits:: + Show only DSOs with hits. +-i:: +--input=:: + Input file name. (default: perf.data unless stdin is a fifo) +-f:: +--force:: + Don't do ownership validation. +-k:: +--kernel:: + Show running kernel build id. +-v:: +--verbose:: + Be more verbose. + +SEE ALSO +-------- +linkperf:perf-record[1], linkperf:perf-top[1], +linkperf:perf-report[1] diff --git a/tools/perf/Documentation/perf-diff.txt b/tools/perf/Documentation/perf-diff.txt new file mode 100644 index 000000000..5b3123d57 --- /dev/null +++ b/tools/perf/Documentation/perf-diff.txt @@ -0,0 +1,133 @@ +perf-diff(1) +============ + +NAME +---- +perf-diff - Read two perf.data files and display the differential profile + +SYNOPSIS +-------- +[verse] +'perf diff' [oldfile] [newfile] + +DESCRIPTION +----------- +This command displays the performance difference amongst two perf.data files +captured via perf record. + +If no parameters are passed it will assume perf.data.old and perf.data. + +The differential profile is displayed only for events matching both +specified perf.data files. + +OPTIONS +------- +-D:: +--dump-raw-trace:: + Dump raw trace in ASCII. + +-m:: +--modules:: + Load module symbols. WARNING: use only with -k and LIVE kernel + +-d:: +--dsos=:: + Only consider symbols in these dsos. CSV that understands + file://filename entries. + +-C:: +--comms=:: + Only consider symbols in these comms. CSV that understands + file://filename entries. + +-S:: +--symbols=:: + Only consider these symbols. CSV that understands + file://filename entries. + +-s:: +--sort=:: + Sort by key(s): pid, comm, dso, symbol. + +-t:: +--field-separator=:: + + Use a special separator character and don't pad with spaces, replacing + all occurrences of this separator in symbol names (and other output) + with a '.' character, that thus it's the only non valid separator. + +-v:: +--verbose:: + Be verbose, for instance, show the raw counts in addition to the + diff. + +-f:: +--force:: + Don't complain, do it. + +--symfs=:: + Look for files with symbols relative to this directory. + +-b:: +--baseline-only:: + Show only items with match in baseline. + +-c:: +--compute:: + Differential computation selection - delta,ratio,wdiff (default is delta). + If '+' is specified as a first character, the output is sorted based + on the computation results. + See COMPARISON METHODS section for more info. + +-p:: +--period:: + Show period values for both compared hist entries. + +-F:: +--formula:: + Show formula for given computation. + +COMPARISON METHODS +------------------ +delta +~~~~~ +If specified the 'Delta' column is displayed with value 'd' computed as: + + d = A->period_percent - B->period_percent + +with: + - A/B being matching hist entry from first/second file specified + (or perf.data/perf.data.old) respectively. + + - period_percent being the % of the hist entry period value within + single data file + +ratio +~~~~~ +If specified the 'Ratio' column is displayed with value 'r' computed as: + + r = A->period / B->period + +with: + - A/B being matching hist entry from first/second file specified + (or perf.data/perf.data.old) respectively. + + - period being the hist entry period value + +wdiff +~~~~~ +If specified the 'Weighted diff' column is displayed with value 'd' computed as: + + d = B->period * WEIGHT-A - A->period * WEIGHT-B + + - A/B being matching hist entry from first/second file specified + (or perf.data/perf.data.old) respectively. + + - period being the hist entry period value + + - WEIGHT-A/WEIGHT-B being user suplied weights in the the '-c' option + behind ':' separator like '-c wdiff:1,2'. + +SEE ALSO +-------- +linkperf:perf-record[1] diff --git a/tools/perf/Documentation/perf-evlist.txt b/tools/perf/Documentation/perf-evlist.txt new file mode 100644 index 000000000..1ceb3700f --- /dev/null +++ b/tools/perf/Documentation/perf-evlist.txt @@ -0,0 +1,38 @@ +perf-evlist(1) +============== + +NAME +---- +perf-evlist - List the event names in a perf.data file + +SYNOPSIS +-------- +[verse] +'perf evlist ' + +DESCRIPTION +----------- +This command displays the names of events sampled in a perf.data file. + +OPTIONS +------- +-i:: +--input=:: + Input file name. (default: perf.data unless stdin is a fifo) + +-F:: +--freq=:: + Show just the sample frequency used for each event. + +-v:: +--verbose=:: + Show all fields. + +-g:: +--group:: + Show event group information. + +SEE ALSO +-------- +linkperf:perf-record[1], linkperf:perf-list[1], +linkperf:perf-report[1] diff --git a/tools/perf/Documentation/perf-help.txt b/tools/perf/Documentation/perf-help.txt new file mode 100644 index 000000000..514391818 --- /dev/null +++ b/tools/perf/Documentation/perf-help.txt @@ -0,0 +1,38 @@ +perf-help(1) +============ + +NAME +---- +perf-help - display help information about perf + +SYNOPSIS +-------- +'perf help' [-a|--all] [COMMAND] + +DESCRIPTION +----------- + +With no options and no COMMAND given, the synopsis of the 'perf' +command and a list of the most commonly used perf commands are printed +on the standard output. + +If the option '--all' or '-a' is given, then all available commands are +printed on the standard output. + +If a perf command is named, a manual page for that command is brought +up. The 'man' program is used by default for this purpose, but this +can be overridden by other options or configuration variables. + +Note that `perf --help ...` is identical to `perf help ...` because the +former is internally converted into the latter. + +OPTIONS +------- +-a:: +--all:: + Prints all the available commands on the standard output. This + option supersedes any other option. + +PERF +---- +Part of the linkperf:perf[1] suite diff --git a/tools/perf/Documentation/perf-inject.txt b/tools/perf/Documentation/perf-inject.txt new file mode 100644 index 000000000..a00a34276 --- /dev/null +++ b/tools/perf/Documentation/perf-inject.txt @@ -0,0 +1,46 @@ +perf-inject(1) +============== + +NAME +---- +perf-inject - Filter to augment the events stream with additional information + +SYNOPSIS +-------- +[verse] +'perf inject ' + +DESCRIPTION +----------- +perf-inject reads a perf-record event stream and repipes it to stdout. At any +point the processing code can inject other events into the event stream - in +this case build-ids (-b option) are read and injected as needed into the event +stream. + +Build-ids are just the first user of perf-inject - potentially anything that +needs userspace processing to augment the events stream with additional +information could make use of this facility. + +OPTIONS +------- +-b:: +--build-ids=:: + Inject build-ids into the output stream +-v:: +--verbose:: + Be more verbose. +-i:: +--input=:: + Input file name. (default: stdin) +-o:: +--output=:: + Output file name. (default: stdout) +-s:: +--sched-stat:: + Merge sched_stat and sched_switch for getting events where and how long + tasks slept. sched_switch contains a callchain where a task slept and + sched_stat contains a timeslice how long a task slept. + +SEE ALSO +-------- +linkperf:perf-record[1], linkperf:perf-report[1], linkperf:perf-archive[1] diff --git a/tools/perf/Documentation/perf-kmem.txt b/tools/perf/Documentation/perf-kmem.txt new file mode 100644 index 000000000..7c8fbbf3f --- /dev/null +++ b/tools/perf/Documentation/perf-kmem.txt @@ -0,0 +1,47 @@ +perf-kmem(1) +============ + +NAME +---- +perf-kmem - Tool to trace/measure kernel memory(slab) properties + +SYNOPSIS +-------- +[verse] +'perf kmem' {record|stat} [] + +DESCRIPTION +----------- +There are two variants of perf kmem: + + 'perf kmem record ' to record the kmem events + of an arbitrary workload. + + 'perf kmem stat' to report kernel memory statistics. + +OPTIONS +------- +-i :: +--input=:: + Select the input file (default: perf.data unless stdin is a fifo) + +--caller:: + Show per-callsite statistics + +--alloc:: + Show per-allocation statistics + +-s :: +--sort=:: + Sort the output (default: frag,hit,bytes) + +-l :: +--line=:: + Print n lines only + +--raw-ip:: + Print raw ip instead of symbol + +SEE ALSO +-------- +linkperf:perf-record[1] diff --git a/tools/perf/Documentation/perf-kvm.txt b/tools/perf/Documentation/perf-kvm.txt new file mode 100644 index 000000000..326f2cb33 --- /dev/null +++ b/tools/perf/Documentation/perf-kvm.txt @@ -0,0 +1,100 @@ +perf-kvm(1) +=========== + +NAME +---- +perf-kvm - Tool to trace/measure kvm guest os + +SYNOPSIS +-------- +[verse] +'perf kvm' [--host] [--guest] [--guestmount= + [--guestkallsyms= --guestmodules= | --guestvmlinux=]] + {top|record|report|diff|buildid-list} +'perf kvm' [--host] [--guest] [--guestkallsyms= --guestmodules= + | --guestvmlinux=] {top|record|report|diff|buildid-list|stat} + +DESCRIPTION +----------- +There are a couple of variants of perf kvm: + + 'perf kvm [options] top ' to generates and displays + a performance counter profile of guest os in realtime + of an arbitrary workload. + + 'perf kvm record ' to record the performance counter profile + of an arbitrary workload and save it into a perf data file. If both + --host and --guest are input, the perf data file name is perf.data.kvm. + If there is no --host but --guest, the file name is perf.data.guest. + If there is no --guest but --host, the file name is perf.data.host. + + 'perf kvm report' to display the performance counter profile information + recorded via perf kvm record. + + 'perf kvm diff' to displays the performance difference amongst two perf.data + files captured via perf record. + + 'perf kvm buildid-list' to display the buildids found in a perf data file, + so that other tools can be used to fetch packages with matching symbol tables + for use by perf report. + + 'perf kvm stat ' to run a command and gather performance counter + statistics. + Especially, perf 'kvm stat record/report' generates a statistical analysis + of KVM events. Currently, vmexit, mmio and ioport events are supported. + 'perf kvm stat record ' records kvm events and the events between + start and end . + And this command produces a file which contains tracing results of kvm + events. + + 'perf kvm stat report' reports statistical data which includes events + handled time, samples, and so on. + +OPTIONS +------- +-i:: +--input=:: + Input file name. +-o:: +--output:: + Output file name. +--host=:: + Collect host side performance profile. +--guest=:: + Collect guest side performance profile. +--guestmount=:: + Guest os root file system mount directory. Users mounts guest os + root directories under by a specific filesystem access method, + typically, sshfs. For example, start 2 guest os. The one's pid is 8888 + and the other's is 9999. + #mkdir ~/guestmount; cd ~/guestmount + #sshfs -o allow_other,direct_io -p 5551 localhost:/ 8888/ + #sshfs -o allow_other,direct_io -p 5552 localhost:/ 9999/ + #perf kvm --host --guest --guestmount=~/guestmount top +--guestkallsyms=:: + Guest os /proc/kallsyms file copy. 'perf' kvm' reads it to get guest + kernel symbols. Users copy it out from guest os. +--guestmodules=:: + Guest os /proc/modules file copy. 'perf' kvm' reads it to get guest + kernel module information. Users copy it out from guest os. +--guestvmlinux=:: + Guest os kernel vmlinux. + +STAT REPORT OPTIONS +------------------- +--vcpu=:: + analyze events which occures on this vcpu. (default: all vcpus) + +--events=:: + events to be analyzed. Possible values: vmexit, mmio, ioport. + (default: vmexit) +-k:: +--key=:: + Sorting key. Possible values: sample (default, sort by samples + number), time (sort by average time). + +SEE ALSO +-------- +linkperf:perf-top[1], linkperf:perf-record[1], linkperf:perf-report[1], +linkperf:perf-diff[1], linkperf:perf-buildid-list[1], +linkperf:perf-stat[1] diff --git a/tools/perf/Documentation/perf-list.txt b/tools/perf/Documentation/perf-list.txt new file mode 100644 index 000000000..d1e39dc8c --- /dev/null +++ b/tools/perf/Documentation/perf-list.txt @@ -0,0 +1,118 @@ +perf-list(1) +============ + +NAME +---- +perf-list - List all symbolic event types + +SYNOPSIS +-------- +[verse] +'perf list' [hw|sw|cache|tracepoint|event_glob] + +DESCRIPTION +----------- +This command displays the symbolic event types which can be selected in the +various perf commands with the -e option. + +[[EVENT_MODIFIERS]] +EVENT MODIFIERS +--------------- + +Events can optionally have a modifer by appending a colon and one or +more modifiers. Modifiers allow the user to restrict the events to be +counted. The following modifiers exist: + + u - user-space counting + k - kernel counting + h - hypervisor counting + G - guest counting (in KVM guests) + H - host counting (not in KVM guests) + p - precise level + +The 'p' modifier can be used for specifying how precise the instruction +address should be. The 'p' modifier can be specified multiple times: + + 0 - SAMPLE_IP can have arbitrary skid + 1 - SAMPLE_IP must have constant skid + 2 - SAMPLE_IP requested to have 0 skid + 3 - SAMPLE_IP must have 0 skid + +For Intel systems precise event sampling is implemented with PEBS +which supports up to precise-level 2. + +On AMD systems it is implemented using IBS (up to precise-level 2). +The precise modifier works with event types 0x76 (cpu-cycles, CPU +clocks not halted) and 0xC1 (micro-ops retired). Both events map to +IBS execution sampling (IBS op) with the IBS Op Counter Control bit +(IbsOpCntCtl) set respectively (see AMD64 Architecture Programmer’s +Manual Volume 2: System Programming, 13.3 Instruction-Based +Sampling). Examples to use IBS: + + perf record -a -e cpu-cycles:p ... # use ibs op counting cycles + perf record -a -e r076:p ... # same as -e cpu-cycles:p + perf record -a -e r0C1:p ... # use ibs op counting micro-ops + +RAW HARDWARE EVENT DESCRIPTOR +----------------------------- +Even when an event is not available in a symbolic form within perf right now, +it can be encoded in a per processor specific way. + +For instance For x86 CPUs NNN represents the raw register encoding with the +layout of IA32_PERFEVTSELx MSRs (see [Intel® 64 and IA-32 Architectures Software Developer's Manual Volume 3B: System Programming Guide] Figure 30-1 Layout +of IA32_PERFEVTSELx MSRs) or AMD's PerfEvtSeln (see [AMD64 Architecture Programmer’s Manual Volume 2: System Programming], Page 344, +Figure 13-7 Performance Event-Select Register (PerfEvtSeln)). + +Note: Only the following bit fields can be set in x86 counter +registers: event, umask, edge, inv, cmask. Esp. guest/host only and +OS/user mode flags must be setup using <>. + +Example: + +If the Intel docs for a QM720 Core i7 describe an event as: + + Event Umask Event Mask + Num. Value Mnemonic Description Comment + + A8H 01H LSD.UOPS Counts the number of micro-ops Use cmask=1 and + delivered by loop stream detector invert to count + cycles + +raw encoding of 0x1A8 can be used: + + perf stat -e r1a8 -a sleep 1 + perf record -e r1a8 ... + +You should refer to the processor specific documentation for getting these +details. Some of them are referenced in the SEE ALSO section below. + +OPTIONS +------- + +Without options all known events will be listed. + +To limit the list use: + +. 'hw' or 'hardware' to list hardware events such as cache-misses, etc. + +. 'sw' or 'software' to list software events such as context switches, etc. + +. 'cache' or 'hwcache' to list hardware cache events such as L1-dcache-loads, etc. + +. 'tracepoint' to list all tracepoint events, alternatively use + 'subsys_glob:event_glob' to filter by tracepoint subsystems such as sched, + block, etc. + +. If none of the above is matched, it will apply the supplied glob to all + events, printing the ones that match. + +One or more types can be used at the same time, listing the events for the +types specified. + +SEE ALSO +-------- +linkperf:perf-stat[1], linkperf:perf-top[1], +linkperf:perf-record[1], +http://www.intel.com/Assets/PDF/manual/253669.pdf[Intel® 64 and IA-32 Architectures Software Developer's Manual Volume 3B: System Programming Guide], +http://support.amd.com/us/Processor_TechDocs/24593_APM_v2.pdf[AMD64 Architecture Programmer’s Manual Volume 2: System Programming] diff --git a/tools/perf/Documentation/perf-lock.txt b/tools/perf/Documentation/perf-lock.txt new file mode 100644 index 000000000..c7f5f5563 --- /dev/null +++ b/tools/perf/Documentation/perf-lock.txt @@ -0,0 +1,66 @@ +perf-lock(1) +============ + +NAME +---- +perf-lock - Analyze lock events + +SYNOPSIS +-------- +[verse] +'perf lock' {record|report|script|info} + +DESCRIPTION +----------- +You can analyze various lock behaviours +and statistics with this 'perf lock' command. + + 'perf lock record ' records lock events + between start and end . And this command + produces the file "perf.data" which contains tracing + results of lock events. + + 'perf lock report' reports statistical data. + + 'perf lock script' shows raw lock events. + + 'perf lock info' shows metadata like threads or addresses + of lock instances. + +COMMON OPTIONS +-------------- + +-i:: +--input=:: + Input file name. (default: perf.data unless stdin is a fifo) + +-v:: +--verbose:: + Be more verbose (show symbol address, etc). + +-D:: +--dump-raw-trace:: + Dump raw trace in ASCII. + +REPORT OPTIONS +-------------- + +-k:: +--key=:: + Sorting key. Possible values: acquired (default), contended, + wait_total, wait_max, wait_min. + +INFO OPTIONS +------------ + +-t:: +--threads:: + dump thread list in perf.data + +-m:: +--map:: + dump map of lock instances (address:name table) + +SEE ALSO +-------- +linkperf:perf[1] diff --git a/tools/perf/Documentation/perf-mem.txt b/tools/perf/Documentation/perf-mem.txt new file mode 100644 index 000000000..888d51137 --- /dev/null +++ b/tools/perf/Documentation/perf-mem.txt @@ -0,0 +1,48 @@ +perf-mem(1) +=========== + +NAME +---- +perf-mem - Profile memory accesses + +SYNOPSIS +-------- +[verse] +'perf mem' [] (record [] | report) + +DESCRIPTION +----------- +"perf mem -t record" runs a command and gathers memory operation data +from it, into perf.data. Perf record options are accepted and are passed through. + +"perf mem -t report" displays the result. It invokes perf report with the +right set of options to display a memory access profile. + +OPTIONS +------- +...:: + Any command you can specify in a shell. + +-t:: +--type=:: + Select the memory operation type: load or store (default: load) + +-D:: +--dump-raw-samples=:: + Dump the raw decoded samples on the screen in a format that is easy to parse with + one sample per line. + +-x:: +--field-separator:: + Specify the field separator used when dump raw samples (-D option). By default, + The separator is the space character. + +-C:: +--cpu-list:: + Restrict dump of raw samples to those provided via this option. Note that the same + option can be passed in record mode. It will be interpreted the same way as perf + record. + +SEE ALSO +-------- +linkperf:perf-record[1], linkperf:perf-report[1] diff --git a/tools/perf/Documentation/perf-probe.txt b/tools/perf/Documentation/perf-probe.txt new file mode 100644 index 000000000..b715cb715 --- /dev/null +++ b/tools/perf/Documentation/perf-probe.txt @@ -0,0 +1,205 @@ +perf-probe(1) +============= + +NAME +---- +perf-probe - Define new dynamic tracepoints + +SYNOPSIS +-------- +[verse] +'perf probe' [options] --add='PROBE' [...] +or +'perf probe' [options] PROBE +or +'perf probe' [options] --del='[GROUP:]EVENT' [...] +or +'perf probe' --list +or +'perf probe' [options] --line='LINE' +or +'perf probe' [options] --vars='PROBEPOINT' + +DESCRIPTION +----------- +This command defines dynamic tracepoint events, by symbol and registers +without debuginfo, or by C expressions (C line numbers, C function names, +and C local variables) with debuginfo. + + +OPTIONS +------- +-k:: +--vmlinux=PATH:: + Specify vmlinux path which has debuginfo (Dwarf binary). + +-m:: +--module=MODNAME|PATH:: + Specify module name in which perf-probe searches probe points + or lines. If a path of module file is passed, perf-probe + treat it as an offline module (this means you can add a probe on + a module which has not been loaded yet). + +-s:: +--source=PATH:: + Specify path to kernel source. + +-v:: +--verbose:: + Be more verbose (show parsed arguments, etc). + +-a:: +--add=:: + Define a probe event (see PROBE SYNTAX for detail). + +-d:: +--del=:: + Delete probe events. This accepts glob wildcards('*', '?') and character + classes(e.g. [a-z], [!A-Z]). + +-l:: +--list:: + List up current probe events. + +-L:: +--line=:: + Show source code lines which can be probed. This needs an argument + which specifies a range of the source code. (see LINE SYNTAX for detail) + +-V:: +--vars=:: + Show available local variables at given probe point. The argument + syntax is same as PROBE SYNTAX, but NO ARGs. + +--externs:: + (Only for --vars) Show external defined variables in addition to local + variables. + +-F:: +--funcs:: + Show available functions in given module or kernel. With -x/--exec, + can also list functions in a user space executable / shared library. + +--filter=FILTER:: + (Only for --vars and --funcs) Set filter. FILTER is a combination of glob + pattern, see FILTER PATTERN for detail. + Default FILTER is "!__k???tab_* & !__crc_*" for --vars, and "!_*" + for --funcs. + If several filters are specified, only the last filter is used. + +-f:: +--force:: + Forcibly add events with existing name. + +-n:: +--dry-run:: + Dry run. With this option, --add and --del doesn't execute actual + adding and removal operations. + +--max-probes:: + Set the maximum number of probe points for an event. Default is 128. + +-x:: +--exec=PATH:: + Specify path to the executable or shared library file for user + space tracing. Can also be used with --funcs option. + +In absence of -m/-x options, perf probe checks if the first argument after +the options is an absolute path name. If its an absolute path, perf probe +uses it as a target module/target user space binary to probe. + +PROBE SYNTAX +------------ +Probe points are defined by following syntax. + + 1) Define event based on function name + [EVENT=]FUNC[@SRC][:RLN|+OFFS|%return|;PTN] [ARG ...] + + 2) Define event based on source file with line number + [EVENT=]SRC:ALN [ARG ...] + + 3) Define event based on source file with lazy pattern + [EVENT=]SRC;PTN [ARG ...] + + +'EVENT' specifies the name of new event, if omitted, it will be set the name of the probed function. Currently, event group name is set as 'probe'. +'FUNC' specifies a probed function name, and it may have one of the following options; '+OFFS' is the offset from function entry address in bytes, ':RLN' is the relative-line number from function entry line, and '%return' means that it probes function return. And ';PTN' means lazy matching pattern (see LAZY MATCHING). Note that ';PTN' must be the end of the probe point definition. In addition, '@SRC' specifies a source file which has that function. +It is also possible to specify a probe point by the source line number or lazy matching by using 'SRC:ALN' or 'SRC;PTN' syntax, where 'SRC' is the source file path, ':ALN' is the line number and ';PTN' is the lazy matching pattern. +'ARG' specifies the arguments of this probe point, (see PROBE ARGUMENT). + +PROBE ARGUMENT +-------------- +Each probe argument follows below syntax. + + [NAME=]LOCALVAR|$retval|%REG|@SYMBOL[:TYPE] + +'NAME' specifies the name of this argument (optional). You can use the name of local variable, local data structure member (e.g. var->field, var.field2), local array with fixed index (e.g. array[1], var->array[0], var->pointer[2]), or kprobe-tracer argument format (e.g. $retval, %ax, etc). Note that the name of this argument will be set as the last member name if you specify a local data structure member (e.g. field2 for 'var->field1.field2'.) +'TYPE' casts the type of this argument (optional). If omitted, perf probe automatically set the type based on debuginfo. You can specify 'string' type only for the local variable or structure member which is an array of or a pointer to 'char' or 'unsigned char' type. + +LINE SYNTAX +----------- +Line range is described by following syntax. + + "FUNC[@SRC][:RLN[+NUM|-RLN2]]|SRC[:ALN[+NUM|-ALN2]]" + +FUNC specifies the function name of showing lines. 'RLN' is the start line +number from function entry line, and 'RLN2' is the end line number. As same as +probe syntax, 'SRC' means the source file path, 'ALN' is start line number, +and 'ALN2' is end line number in the file. It is also possible to specify how +many lines to show by using 'NUM'. Moreover, 'FUNC@SRC' combination is good +for searching a specific function when several functions share same name. +So, "source.c:100-120" shows lines between 100th to l20th in source.c file. And "func:10+20" shows 20 lines from 10th line of func function. + +LAZY MATCHING +------------- + The lazy line matching is similar to glob matching but ignoring spaces in both of pattern and target. So this accepts wildcards('*', '?') and character classes(e.g. [a-z], [!A-Z]). + +e.g. + 'a=*' can matches 'a=b', 'a = b', 'a == b' and so on. + +This provides some sort of flexibility and robustness to probe point definitions against minor code changes. For example, actual 10th line of schedule() can be moved easily by modifying schedule(), but the same line matching 'rq=cpu_rq*' may still exist in the function.) + +FILTER PATTERN +-------------- + The filter pattern is a glob matching pattern(s) to filter variables. + In addition, you can use "!" for specifying filter-out rule. You also can give several rules combined with "&" or "|", and fold those rules as one rule by using "(" ")". + +e.g. + With --filter "foo* | bar*", perf probe -V shows variables which start with "foo" or "bar". + With --filter "!foo* & *bar", perf probe -V shows variables which don't start with "foo" and end with "bar", like "fizzbar". But "foobar" is filtered out. + +EXAMPLES +-------- +Display which lines in schedule() can be probed: + + ./perf probe --line schedule + +Add a probe on schedule() function 12th line with recording cpu local variable: + + ./perf probe schedule:12 cpu + or + ./perf probe --add='schedule:12 cpu' + + this will add one or more probes which has the name start with "schedule". + + Add probes on lines in schedule() function which calls update_rq_clock(). + + ./perf probe 'schedule;update_rq_clock*' + or + ./perf probe --add='schedule;update_rq_clock*' + +Delete all probes on schedule(). + + ./perf probe --del='schedule*' + +Add probes at zfree() function on /bin/zsh + + ./perf probe -x /bin/zsh zfree or ./perf probe /bin/zsh zfree + +Add probes at malloc() function on libc + + ./perf probe -x /lib/libc.so.6 malloc or ./perf probe /lib/libc.so.6 malloc + +SEE ALSO +-------- +linkperf:perf-trace[1], linkperf:perf-record[1] diff --git a/tools/perf/Documentation/perf-record.txt b/tools/perf/Documentation/perf-record.txt new file mode 100644 index 000000000..d4da111ef --- /dev/null +++ b/tools/perf/Documentation/perf-record.txt @@ -0,0 +1,193 @@ +perf-record(1) +============== + +NAME +---- +perf-record - Run a command and record its profile into perf.data + +SYNOPSIS +-------- +[verse] +'perf record' [-e | --event=EVENT] [-l] [-a] +'perf record' [-e | --event=EVENT] [-l] [-a] -- [] + +DESCRIPTION +----------- +This command runs a command and gathers a performance counter profile +from it, into perf.data - without displaying anything. + +This file can then be inspected later on, using 'perf report'. + + +OPTIONS +------- +...:: + Any command you can specify in a shell. + +-e:: +--event=:: + Select the PMU event. Selection can be: + + - a symbolic event name (use 'perf list' to list all events) + + - a raw PMU event (eventsel+umask) in the form of rNNN where NNN is a + hexadecimal event descriptor. + + - a hardware breakpoint event in the form of '\mem:addr[:access]' + where addr is the address in memory you want to break in. + Access is the memory access type (read, write, execute) it can + be passed as follows: '\mem:addr[:[r][w][x]]'. + If you want to profile read-write accesses in 0x1000, just set + 'mem:0x1000:rw'. + +--filter=:: + Event filter. + +-a:: +--all-cpus:: + System-wide collection from all CPUs. + +-l:: + Scale counter values. + +-p:: +--pid=:: + Record events on existing process ID (comma separated list). + +-t:: +--tid=:: + Record events on existing thread ID (comma separated list). + +-u:: +--uid=:: + Record events in threads owned by uid. Name or number. + +-r:: +--realtime=:: + Collect data with this RT SCHED_FIFO priority. +-D:: +--no-delay:: + Collect data without buffering. +-A:: +--append:: + Append to the output file to do incremental profiling. + +-f:: +--force:: + Overwrite existing data file. (deprecated) + +-c:: +--count=:: + Event period to sample. + +-o:: +--output=:: + Output file name. + +-i:: +--no-inherit:: + Child tasks do not inherit counters. +-F:: +--freq=:: + Profile at this frequency. + +-m:: +--mmap-pages=:: + Number of mmap data pages. Must be a power of two. + +-g:: +--call-graph:: + Do call-graph (stack chain/backtrace) recording. + +-q:: +--quiet:: + Don't print any message, useful for scripting. + +-v:: +--verbose:: + Be more verbose (show counter open errors, etc). + +-s:: +--stat:: + Per thread counts. + +-d:: +--data:: + Sample addresses. + +-T:: +--timestamp:: + Sample timestamps. Use it with 'perf report -D' to see the timestamps, + for instance. + +-n:: +--no-samples:: + Don't sample. + +-R:: +--raw-samples:: +Collect raw sample records from all opened counters (default for tracepoint counters). + +-C:: +--cpu:: +Collect samples only on the list of CPUs provided. Multiple CPUs can be provided as a +comma-separated list with no space: 0,1. Ranges of CPUs are specified with -: 0-2. +In per-thread mode with inheritance mode on (default), samples are captured only when +the thread executes on the designated CPUs. Default is to monitor all CPUs. + +-N:: +--no-buildid-cache:: +Do not update the builid cache. This saves some overhead in situations +where the information in the perf.data file (which includes buildids) +is sufficient. + +-G name,...:: +--cgroup name,...:: +monitor only in the container (cgroup) called "name". This option is available only +in per-cpu mode. The cgroup filesystem must be mounted. All threads belonging to +container "name" are monitored when they run on the monitored CPUs. Multiple cgroups +can be provided. Each cgroup is applied to the corresponding event, i.e., first cgroup +to first event, second cgroup to second event and so on. It is possible to provide +an empty cgroup (monitor all the time) using, e.g., -G foo,,bar. Cgroups must have +corresponding events, i.e., they always refer to events defined earlier on the command +line. + +-b:: +--branch-any:: +Enable taken branch stack sampling. Any type of taken branch may be sampled. +This is a shortcut for --branch-filter any. See --branch-filter for more infos. + +-j:: +--branch-filter:: +Enable taken branch stack sampling. Each sample captures a series of consecutive +taken branches. The number of branches captured with each sample depends on the +underlying hardware, the type of branches of interest, and the executed code. +It is possible to select the types of branches captured by enabling filters. The +following filters are defined: + + - any: any type of branches + - any_call: any function call or system call + - any_ret: any function return or system call return + - ind_call: any indirect branch + - u: only when the branch target is at the user level + - k: only when the branch target is in the kernel + - hv: only when the target is at the hypervisor level + ++ +The option requires at least one branch type among any, any_call, any_ret, ind_call. +The privilege levels may be omitted, in which case, the privilege levels of the associated +event are applied to the branch filter. Both kernel (k) and hypervisor (hv) privilege +levels are subject to permissions. When sampling on multiple events, branch stack sampling +is enabled for all the sampling events. The sampled branch type is the same for all events. +The various filters must be specified as a comma separated list: --branch-filter any_ret,u,k +Note that this feature may not be available on all processors. + +-W:: +--weight:: +Enable weightened sampling. An additional weight is recorded per sample and can be +displayed with the weight and local_weight sort keys. This currently works for TSX +abort events and some memory events in precise mode on modern Intel CPUs. + +SEE ALSO +-------- +linkperf:perf-stat[1], linkperf:perf-list[1] diff --git a/tools/perf/Documentation/perf-report.txt b/tools/perf/Documentation/perf-report.txt new file mode 100644 index 000000000..7d5f4f38a --- /dev/null +++ b/tools/perf/Documentation/perf-report.txt @@ -0,0 +1,215 @@ +perf-report(1) +============== + +NAME +---- +perf-report - Read perf.data (created by perf record) and display the profile + +SYNOPSIS +-------- +[verse] +'perf report' [-i | --input=file] + +DESCRIPTION +----------- +This command displays the performance counter profile information recorded +via perf record. + +OPTIONS +------- +-i:: +--input=:: + Input file name. (default: perf.data unless stdin is a fifo) + +-v:: +--verbose:: + Be more verbose. (show symbol address, etc) + +-d:: +--dsos=:: + Only consider symbols in these dsos. CSV that understands + file://filename entries. +-n:: +--show-nr-samples:: + Show the number of samples for each symbol + +--showcpuutilization:: + Show sample percentage for different cpu modes. + +-T:: +--threads:: + Show per-thread event counters +-c:: +--comms=:: + Only consider symbols in these comms. CSV that understands + file://filename entries. +-S:: +--symbols=:: + Only consider these symbols. CSV that understands + file://filename entries. + +--symbol-filter=:: + Only show symbols that match (partially) with this filter. + +-U:: +--hide-unresolved:: + Only display entries resolved to a symbol. + +-s:: +--sort=:: + Sort histogram entries by given key(s) - multiple keys can be specified + in CSV format. Following sort keys are available: + pid, comm, dso, symbol, parent, cpu, srcline, weight, local_weight. + + Each key has following meaning: + + - comm: command (name) of the task which can be read via /proc//comm + - pid: command and tid of the task + - dso: name of library or module executed at the time of sample + - symbol: name of function executed at the time of sample + - parent: name of function matched to the parent regex filter. Unmatched + entries are displayed as "[other]". + - cpu: cpu number the task ran at the time of sample + - srcline: filename and line number executed at the time of sample. The + DWARF debuggin info must be provided. + + By default, comm, dso and symbol keys are used. + (i.e. --sort comm,dso,symbol) + + If --branch-stack option is used, following sort keys are also + available: + dso_from, dso_to, symbol_from, symbol_to, mispredict. + + - dso_from: name of library or module branched from + - dso_to: name of library or module branched to + - symbol_from: name of function branched from + - symbol_to: name of function branched to + - mispredict: "N" for predicted branch, "Y" for mispredicted branch + + And default sort keys are changed to comm, dso_from, symbol_from, dso_to + and symbol_to, see '--branch-stack'. + +-p:: +--parent=:: + A regex filter to identify parent. The parent is a caller of this + function and searched through the callchain, thus it requires callchain + information recorded. The pattern is in the exteneded regex format and + defaults to "\^sys_|^do_page_fault", see '--sort parent'. + +-x:: +--exclude-other:: + Only display entries with parent-match. + +-w:: +--column-widths=:: + Force each column width to the provided list, for large terminal + readability. + +-t:: +--field-separator=:: + Use a special separator character and don't pad with spaces, replacing + all occurrences of this separator in symbol names (and other output) + with a '.' character, that thus it's the only non valid separator. + +-D:: +--dump-raw-trace:: + Dump raw trace in ASCII. + +-g [type,min[,limit],order]:: +--call-graph:: + Display call chains using type, min percent threshold, optional print + limit and order. + type can be either: + - flat: single column, linear exposure of call chains. + - graph: use a graph tree, displaying absolute overhead rates. + - fractal: like graph, but displays relative rates. Each branch of + the tree is considered as a new profiled object. + + + order can be either: + - callee: callee based call graph. + - caller: inverted caller based call graph. + + Default: fractal,0.5,callee. + +-G:: +--inverted:: + alias for inverted caller based call graph. + +--pretty=:: + Pretty printing style. key: normal, raw + +--stdio:: Use the stdio interface. + +--tui:: Use the TUI interface, that is integrated with annotate and allows + zooming into DSOs or threads, among other features. Use of --tui + requires a tty, if one is not present, as when piping to other + commands, the stdio interface is used. + +--gtk:: Use the GTK2 interface. + +-k:: +--vmlinux=:: + vmlinux pathname + +--kallsyms=:: + kallsyms pathname + +-m:: +--modules:: + Load module symbols. WARNING: This should only be used with -k and + a LIVE kernel. + +-f:: +--force:: + Don't complain, do it. + +--symfs=:: + Look for files with symbols relative to this directory. + +-C:: +--cpu:: Only report samples for the list of CPUs provided. Multiple CPUs can + be provided as a comma-separated list with no space: 0,1. Ranges of + CPUs are specified with -: 0-2. Default is to report samples on all + CPUs. + +-M:: +--disassembler-style=:: Set disassembler style for objdump. + +--source:: + Interleave source code with assembly code. Enabled by default, + disable with --no-source. + +--asm-raw:: + Show raw instruction encoding of assembly instructions. + +--show-total-period:: Show a column with the sum of periods. + +-I:: +--show-info:: + Display extended information about the perf.data file. This adds + information which may be very large and thus may clutter the display. + It currently includes: cpu and numa topology of the host system. + +-b:: +--branch-stack:: + Use the addresses of sampled taken branches instead of the instruction + address to build the histograms. To generate meaningful output, the + perf.data file must have been obtained using perf record -b or + perf record --branch-filter xxx where xxx is a branch filter option. + perf report is able to auto-detect whether a perf.data file contains + branch stacks and it will automatically switch to the branch view mode, + unless --no-branch-stack is used. + +--objdump=:: + Path to objdump binary. + +--group:: + Show event group information together. + +--demangle:: + Demangle symbol names to human readable form. It's enabled by default, + disable with --no-demangle. + +SEE ALSO +-------- +linkperf:perf-stat[1], linkperf:perf-annotate[1] diff --git a/tools/perf/Documentation/perf-sched.txt b/tools/perf/Documentation/perf-sched.txt new file mode 100644 index 000000000..8ff4df956 --- /dev/null +++ b/tools/perf/Documentation/perf-sched.txt @@ -0,0 +1,55 @@ +perf-sched(1) +============== + +NAME +---- +perf-sched - Tool to trace/measure scheduler properties (latencies) + +SYNOPSIS +-------- +[verse] +'perf sched' {record|latency|map|replay|script} + +DESCRIPTION +----------- +There are five variants of perf sched: + + 'perf sched record ' to record the scheduling events + of an arbitrary workload. + + 'perf sched latency' to report the per task scheduling latencies + and other scheduling properties of the workload. + + 'perf sched script' to see a detailed trace of the workload that + was recorded (aliased to 'perf script' for now). + + 'perf sched replay' to simulate the workload that was recorded + via perf sched record. (this is done by starting up mockup threads + that mimic the workload based on the events in the trace. These + threads can then replay the timings (CPU runtime and sleep patterns) + of the workload as it occurred when it was recorded - and can repeat + it a number of times, measuring its performance.) + + 'perf sched map' to print a textual context-switching outline of + workload captured via perf sched record. Columns stand for + individual CPUs, and the two-letter shortcuts stand for tasks that + are running on a CPU. A '*' denotes the CPU that had the event, and + a dot signals an idle CPU. + +OPTIONS +------- +-i:: +--input=:: + Input file name. (default: perf.data unless stdin is a fifo) + +-v:: +--verbose:: + Be more verbose. (show symbol address, etc) + +-D:: +--dump-raw-trace=:: + Display verbose dump of the sched data. + +SEE ALSO +-------- +linkperf:perf-record[1] diff --git a/tools/perf/Documentation/perf-script-perl.txt b/tools/perf/Documentation/perf-script-perl.txt new file mode 100644 index 000000000..d00bef231 --- /dev/null +++ b/tools/perf/Documentation/perf-script-perl.txt @@ -0,0 +1,216 @@ +perf-script-perl(1) +================== + +NAME +---- +perf-script-perl - Process trace data with a Perl script + +SYNOPSIS +-------- +[verse] +'perf script' [-s [Perl]:script[.pl] ] + +DESCRIPTION +----------- + +This perf script option is used to process perf script data using perf's +built-in Perl interpreter. It reads and processes the input file and +displays the results of the trace analysis implemented in the given +Perl script, if any. + +STARTER SCRIPTS +--------------- + +You can avoid reading the rest of this document by running 'perf script +-g perl' in the same directory as an existing perf.data trace file. +That will generate a starter script containing a handler for each of +the event types in the trace file; it simply prints every available +field for each event in the trace file. + +You can also look at the existing scripts in +~/libexec/perf-core/scripts/perl for typical examples showing how to +do basic things like aggregate event data, print results, etc. Also, +the check-perf-script.pl script, while not interesting for its results, +attempts to exercise all of the main scripting features. + +EVENT HANDLERS +-------------- + +When perf script is invoked using a trace script, a user-defined +'handler function' is called for each event in the trace. If there's +no handler function defined for a given event type, the event is +ignored (or passed to a 'trace_handled' function, see below) and the +next event is processed. + +Most of the event's field values are passed as arguments to the +handler function; some of the less common ones aren't - those are +available as calls back into the perf executable (see below). + +As an example, the following perf record command can be used to record +all sched_wakeup events in the system: + + # perf record -a -e sched:sched_wakeup + +Traces meant to be processed using a script should be recorded with +the above option: -a to enable system-wide collection. + +The format file for the sched_wakep event defines the following fields +(see /sys/kernel/debug/tracing/events/sched/sched_wakeup/format): + +---- + format: + field:unsigned short common_type; + field:unsigned char common_flags; + field:unsigned char common_preempt_count; + field:int common_pid; + + field:char comm[TASK_COMM_LEN]; + field:pid_t pid; + field:int prio; + field:int success; + field:int target_cpu; +---- + +The handler function for this event would be defined as: + +---- +sub sched::sched_wakeup +{ + my ($event_name, $context, $common_cpu, $common_secs, + $common_nsecs, $common_pid, $common_comm, + $comm, $pid, $prio, $success, $target_cpu) = @_; +} +---- + +The handler function takes the form subsystem::event_name. + +The $common_* arguments in the handler's argument list are the set of +arguments passed to all event handlers; some of the fields correspond +to the common_* fields in the format file, but some are synthesized, +and some of the common_* fields aren't common enough to to be passed +to every event as arguments but are available as library functions. + +Here's a brief description of each of the invariant event args: + + $event_name the name of the event as text + $context an opaque 'cookie' used in calls back into perf + $common_cpu the cpu the event occurred on + $common_secs the secs portion of the event timestamp + $common_nsecs the nsecs portion of the event timestamp + $common_pid the pid of the current task + $common_comm the name of the current process + +All of the remaining fields in the event's format file have +counterparts as handler function arguments of the same name, as can be +seen in the example above. + +The above provides the basics needed to directly access every field of +every event in a trace, which covers 90% of what you need to know to +write a useful trace script. The sections below cover the rest. + +SCRIPT LAYOUT +------------- + +Every perf script Perl script should start by setting up a Perl module +search path and 'use'ing a few support modules (see module +descriptions below): + +---- + use lib "$ENV{'PERF_EXEC_PATH'}/scripts/perl/Perf-Trace-Util/lib"; + use lib "./Perf-Trace-Util/lib"; + use Perf::Trace::Core; + use Perf::Trace::Context; + use Perf::Trace::Util; +---- + +The rest of the script can contain handler functions and support +functions in any order. + +Aside from the event handler functions discussed above, every script +can implement a set of optional functions: + +*trace_begin*, if defined, is called before any event is processed and +gives scripts a chance to do setup tasks: + +---- + sub trace_begin + { + } +---- + +*trace_end*, if defined, is called after all events have been + processed and gives scripts a chance to do end-of-script tasks, such + as display results: + +---- +sub trace_end +{ +} +---- + +*trace_unhandled*, if defined, is called after for any event that + doesn't have a handler explicitly defined for it. The standard set + of common arguments are passed into it: + +---- +sub trace_unhandled +{ + my ($event_name, $context, $common_cpu, $common_secs, + $common_nsecs, $common_pid, $common_comm) = @_; +} +---- + +The remaining sections provide descriptions of each of the available +built-in perf script Perl modules and their associated functions. + +AVAILABLE MODULES AND FUNCTIONS +------------------------------- + +The following sections describe the functions and variables available +via the various Perf::Trace::* Perl modules. To use the functions and +variables from the given module, add the corresponding 'use +Perf::Trace::XXX' line to your perf script script. + +Perf::Trace::Core Module +~~~~~~~~~~~~~~~~~~~~~~~~ + +These functions provide some essential functions to user scripts. + +The *flag_str* and *symbol_str* functions provide human-readable +strings for flag and symbolic fields. These correspond to the strings +and values parsed from the 'print fmt' fields of the event format +files: + + flag_str($event_name, $field_name, $field_value) - returns the string represention corresponding to $field_value for the flag field $field_name of event $event_name + symbol_str($event_name, $field_name, $field_value) - returns the string represention corresponding to $field_value for the symbolic field $field_name of event $event_name + +Perf::Trace::Context Module +~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Some of the 'common' fields in the event format file aren't all that +common, but need to be made accessible to user scripts nonetheless. + +Perf::Trace::Context defines a set of functions that can be used to +access this data in the context of the current event. Each of these +functions expects a $context variable, which is the same as the +$context variable passed into every event handler as the second +argument. + + common_pc($context) - returns common_preempt count for the current event + common_flags($context) - returns common_flags for the current event + common_lock_depth($context) - returns common_lock_depth for the current event + +Perf::Trace::Util Module +~~~~~~~~~~~~~~~~~~~~~~~~ + +Various utility functions for use with perf script: + + nsecs($secs, $nsecs) - returns total nsecs given secs/nsecs pair + nsecs_secs($nsecs) - returns whole secs portion given nsecs + nsecs_nsecs($nsecs) - returns nsecs remainder given nsecs + nsecs_str($nsecs) - returns printable string in the form secs.nsecs + avg($total, $n) - returns average given a sum and a total number of values + +SEE ALSO +-------- +linkperf:perf-script[1] diff --git a/tools/perf/Documentation/perf-script-python.txt b/tools/perf/Documentation/perf-script-python.txt new file mode 100644 index 000000000..9f1f054b8 --- /dev/null +++ b/tools/perf/Documentation/perf-script-python.txt @@ -0,0 +1,620 @@ +perf-script-python(1) +==================== + +NAME +---- +perf-script-python - Process trace data with a Python script + +SYNOPSIS +-------- +[verse] +'perf script' [-s [Python]:script[.py] ] + +DESCRIPTION +----------- + +This perf script option is used to process perf script data using perf's +built-in Python interpreter. It reads and processes the input file and +displays the results of the trace analysis implemented in the given +Python script, if any. + +A QUICK EXAMPLE +--------------- + +This section shows the process, start to finish, of creating a working +Python script that aggregates and extracts useful information from a +raw perf script stream. You can avoid reading the rest of this +document if an example is enough for you; the rest of the document +provides more details on each step and lists the library functions +available to script writers. + +This example actually details the steps that were used to create the +'syscall-counts' script you see when you list the available perf script +scripts via 'perf script -l'. As such, this script also shows how to +integrate your script into the list of general-purpose 'perf script' +scripts listed by that command. + +The syscall-counts script is a simple script, but demonstrates all the +basic ideas necessary to create a useful script. Here's an example +of its output (syscall names are not yet supported, they will appear +as numbers): + +---- +syscall events: + +event count +---------------------------------------- ----------- +sys_write 455067 +sys_getdents 4072 +sys_close 3037 +sys_swapoff 1769 +sys_read 923 +sys_sched_setparam 826 +sys_open 331 +sys_newfstat 326 +sys_mmap 217 +sys_munmap 216 +sys_futex 141 +sys_select 102 +sys_poll 84 +sys_setitimer 12 +sys_writev 8 +15 8 +sys_lseek 7 +sys_rt_sigprocmask 6 +sys_wait4 3 +sys_ioctl 3 +sys_set_robust_list 1 +sys_exit 1 +56 1 +sys_access 1 +---- + +Basically our task is to keep a per-syscall tally that gets updated +every time a system call occurs in the system. Our script will do +that, but first we need to record the data that will be processed by +that script. Theoretically, there are a couple of ways we could do +that: + +- we could enable every event under the tracing/events/syscalls + directory, but this is over 600 syscalls, well beyond the number + allowable by perf. These individual syscall events will however be + useful if we want to later use the guidance we get from the + general-purpose scripts to drill down and get more detail about + individual syscalls of interest. + +- we can enable the sys_enter and/or sys_exit syscalls found under + tracing/events/raw_syscalls. These are called for all syscalls; the + 'id' field can be used to distinguish between individual syscall + numbers. + +For this script, we only need to know that a syscall was entered; we +don't care how it exited, so we'll use 'perf record' to record only +the sys_enter events: + +---- +# perf record -a -e raw_syscalls:sys_enter + +^C[ perf record: Woken up 1 times to write data ] +[ perf record: Captured and wrote 56.545 MB perf.data (~2470503 samples) ] +---- + +The options basically say to collect data for every syscall event +system-wide and multiplex the per-cpu output into a single stream. +That single stream will be recorded in a file in the current directory +called perf.data. + +Once we have a perf.data file containing our data, we can use the -g +'perf script' option to generate a Python script that will contain a +callback handler for each event type found in the perf.data trace +stream (for more details, see the STARTER SCRIPTS section). + +---- +# perf script -g python +generated Python script: perf-script.py + +The output file created also in the current directory is named +perf-script.py. Here's the file in its entirety: + +# perf script event handlers, generated by perf script -g python +# Licensed under the terms of the GNU GPL License version 2 + +# The common_* event handler fields are the most useful fields common to +# all events. They don't necessarily correspond to the 'common_*' fields +# in the format files. Those fields not available as handler params can +# be retrieved using Python functions of the form common_*(context). +# See the perf-script-python Documentation for the list of available functions. + +import os +import sys + +sys.path.append(os.environ['PERF_EXEC_PATH'] + \ + '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') + +from perf_trace_context import * +from Core import * + +def trace_begin(): + print "in trace_begin" + +def trace_end(): + print "in trace_end" + +def raw_syscalls__sys_enter(event_name, context, common_cpu, + common_secs, common_nsecs, common_pid, common_comm, + id, args): + print_header(event_name, common_cpu, common_secs, common_nsecs, + common_pid, common_comm) + + print "id=%d, args=%s\n" % \ + (id, args), + +def trace_unhandled(event_name, context, common_cpu, common_secs, common_nsecs, + common_pid, common_comm): + print_header(event_name, common_cpu, common_secs, common_nsecs, + common_pid, common_comm) + +def print_header(event_name, cpu, secs, nsecs, pid, comm): + print "%-20s %5u %05u.%09u %8u %-20s " % \ + (event_name, cpu, secs, nsecs, pid, comm), +---- + +At the top is a comment block followed by some import statements and a +path append which every perf script script should include. + +Following that are a couple generated functions, trace_begin() and +trace_end(), which are called at the beginning and the end of the +script respectively (for more details, see the SCRIPT_LAYOUT section +below). + +Following those are the 'event handler' functions generated one for +every event in the 'perf record' output. The handler functions take +the form subsystem__event_name, and contain named parameters, one for +each field in the event; in this case, there's only one event, +raw_syscalls__sys_enter(). (see the EVENT HANDLERS section below for +more info on event handlers). + +The final couple of functions are, like the begin and end functions, +generated for every script. The first, trace_unhandled(), is called +every time the script finds an event in the perf.data file that +doesn't correspond to any event handler in the script. This could +mean either that the record step recorded event types that it wasn't +really interested in, or the script was run against a trace file that +doesn't correspond to the script. + +The script generated by -g option simply prints a line for each +event found in the trace stream i.e. it basically just dumps the event +and its parameter values to stdout. The print_header() function is +simply a utility function used for that purpose. Let's rename the +script and run it to see the default output: + +---- +# mv perf-script.py syscall-counts.py +# perf script -s syscall-counts.py + +raw_syscalls__sys_enter 1 00840.847582083 7506 perf id=1, args= +raw_syscalls__sys_enter 1 00840.847595764 7506 perf id=1, args= +raw_syscalls__sys_enter 1 00840.847620860 7506 perf id=1, args= +raw_syscalls__sys_enter 1 00840.847710478 6533 npviewer.bin id=78, args= +raw_syscalls__sys_enter 1 00840.847719204 6533 npviewer.bin id=142, args= +raw_syscalls__sys_enter 1 00840.847755445 6533 npviewer.bin id=3, args= +raw_syscalls__sys_enter 1 00840.847775601 6533 npviewer.bin id=3, args= +raw_syscalls__sys_enter 1 00840.847781820 6533 npviewer.bin id=3, args= +. +. +. +---- + +Of course, for this script, we're not interested in printing every +trace event, but rather aggregating it in a useful way. So we'll get +rid of everything to do with printing as well as the trace_begin() and +trace_unhandled() functions, which we won't be using. That leaves us +with this minimalistic skeleton: + +---- +import os +import sys + +sys.path.append(os.environ['PERF_EXEC_PATH'] + \ + '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') + +from perf_trace_context import * +from Core import * + +def trace_end(): + print "in trace_end" + +def raw_syscalls__sys_enter(event_name, context, common_cpu, + common_secs, common_nsecs, common_pid, common_comm, + id, args): +---- + +In trace_end(), we'll simply print the results, but first we need to +generate some results to print. To do that we need to have our +sys_enter() handler do the necessary tallying until all events have +been counted. A hash table indexed by syscall id is a good way to +store that information; every time the sys_enter() handler is called, +we simply increment a count associated with that hash entry indexed by +that syscall id: + +---- + syscalls = autodict() + + try: + syscalls[id] += 1 + except TypeError: + syscalls[id] = 1 +---- + +The syscalls 'autodict' object is a special kind of Python dictionary +(implemented in Core.py) that implements Perl's 'autovivifying' hashes +in Python i.e. with autovivifying hashes, you can assign nested hash +values without having to go to the trouble of creating intermediate +levels if they don't exist e.g syscalls[comm][pid][id] = 1 will create +the intermediate hash levels and finally assign the value 1 to the +hash entry for 'id' (because the value being assigned isn't a hash +object itself, the initial value is assigned in the TypeError +exception. Well, there may be a better way to do this in Python but +that's what works for now). + +Putting that code into the raw_syscalls__sys_enter() handler, we +effectively end up with a single-level dictionary keyed on syscall id +and having the counts we've tallied as values. + +The print_syscall_totals() function iterates over the entries in the +dictionary and displays a line for each entry containing the syscall +name (the dictonary keys contain the syscall ids, which are passed to +the Util function syscall_name(), which translates the raw syscall +numbers to the corresponding syscall name strings). The output is +displayed after all the events in the trace have been processed, by +calling the print_syscall_totals() function from the trace_end() +handler called at the end of script processing. + +The final script producing the output shown above is shown in its +entirety below (syscall_name() helper is not yet available, you can +only deal with id's for now): + +---- +import os +import sys + +sys.path.append(os.environ['PERF_EXEC_PATH'] + \ + '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') + +from perf_trace_context import * +from Core import * +from Util import * + +syscalls = autodict() + +def trace_end(): + print_syscall_totals() + +def raw_syscalls__sys_enter(event_name, context, common_cpu, + common_secs, common_nsecs, common_pid, common_comm, + id, args): + try: + syscalls[id] += 1 + except TypeError: + syscalls[id] = 1 + +def print_syscall_totals(): + if for_comm is not None: + print "\nsyscall events for %s:\n\n" % (for_comm), + else: + print "\nsyscall events:\n\n", + + print "%-40s %10s\n" % ("event", "count"), + print "%-40s %10s\n" % ("----------------------------------------", \ + "-----------"), + + for id, val in sorted(syscalls.iteritems(), key = lambda(k, v): (v, k), \ + reverse = True): + print "%-40s %10d\n" % (syscall_name(id), val), +---- + +The script can be run just as before: + + # perf script -s syscall-counts.py + +So those are the essential steps in writing and running a script. The +process can be generalized to any tracepoint or set of tracepoints +you're interested in - basically find the tracepoint(s) you're +interested in by looking at the list of available events shown by +'perf list' and/or look in /sys/kernel/debug/tracing events for +detailed event and field info, record the corresponding trace data +using 'perf record', passing it the list of interesting events, +generate a skeleton script using 'perf script -g python' and modify the +code to aggregate and display it for your particular needs. + +After you've done that you may end up with a general-purpose script +that you want to keep around and have available for future use. By +writing a couple of very simple shell scripts and putting them in the +right place, you can have your script listed alongside the other +scripts listed by the 'perf script -l' command e.g.: + +---- +root@tropicana:~# perf script -l +List of available trace scripts: + wakeup-latency system-wide min/max/avg wakeup latency + rw-by-file r/w activity for a program, by file + rw-by-pid system-wide r/w activity +---- + +A nice side effect of doing this is that you also then capture the +probably lengthy 'perf record' command needed to record the events for +the script. + +To have the script appear as a 'built-in' script, you write two simple +scripts, one for recording and one for 'reporting'. + +The 'record' script is a shell script with the same base name as your +script, but with -record appended. The shell script should be put +into the perf/scripts/python/bin directory in the kernel source tree. +In that script, you write the 'perf record' command-line needed for +your script: + +---- +# cat kernel-source/tools/perf/scripts/python/bin/syscall-counts-record + +#!/bin/bash +perf record -a -e raw_syscalls:sys_enter +---- + +The 'report' script is also a shell script with the same base name as +your script, but with -report appended. It should also be located in +the perf/scripts/python/bin directory. In that script, you write the +'perf script -s' command-line needed for running your script: + +---- +# cat kernel-source/tools/perf/scripts/python/bin/syscall-counts-report + +#!/bin/bash +# description: system-wide syscall counts +perf script -s ~/libexec/perf-core/scripts/python/syscall-counts.py +---- + +Note that the location of the Python script given in the shell script +is in the libexec/perf-core/scripts/python directory - this is where +the script will be copied by 'make install' when you install perf. +For the installation to install your script there, your script needs +to be located in the perf/scripts/python directory in the kernel +source tree: + +---- +# ls -al kernel-source/tools/perf/scripts/python + +root@tropicana:/home/trz/src/tip# ls -al tools/perf/scripts/python +total 32 +drwxr-xr-x 4 trz trz 4096 2010-01-26 22:30 . +drwxr-xr-x 4 trz trz 4096 2010-01-26 22:29 .. +drwxr-xr-x 2 trz trz 4096 2010-01-26 22:29 bin +-rw-r--r-- 1 trz trz 2548 2010-01-26 22:29 check-perf-script.py +drwxr-xr-x 3 trz trz 4096 2010-01-26 22:49 Perf-Trace-Util +-rw-r--r-- 1 trz trz 1462 2010-01-26 22:30 syscall-counts.py +---- + +Once you've done that (don't forget to do a new 'make install', +otherwise your script won't show up at run-time), 'perf script -l' +should show a new entry for your script: + +---- +root@tropicana:~# perf script -l +List of available trace scripts: + wakeup-latency system-wide min/max/avg wakeup latency + rw-by-file r/w activity for a program, by file + rw-by-pid system-wide r/w activity + syscall-counts system-wide syscall counts +---- + +You can now perform the record step via 'perf script record': + + # perf script record syscall-counts + +and display the output using 'perf script report': + + # perf script report syscall-counts + +STARTER SCRIPTS +--------------- + +You can quickly get started writing a script for a particular set of +trace data by generating a skeleton script using 'perf script -g +python' in the same directory as an existing perf.data trace file. +That will generate a starter script containing a handler for each of +the event types in the trace file; it simply prints every available +field for each event in the trace file. + +You can also look at the existing scripts in +~/libexec/perf-core/scripts/python for typical examples showing how to +do basic things like aggregate event data, print results, etc. Also, +the check-perf-script.py script, while not interesting for its results, +attempts to exercise all of the main scripting features. + +EVENT HANDLERS +-------------- + +When perf script is invoked using a trace script, a user-defined +'handler function' is called for each event in the trace. If there's +no handler function defined for a given event type, the event is +ignored (or passed to a 'trace_handled' function, see below) and the +next event is processed. + +Most of the event's field values are passed as arguments to the +handler function; some of the less common ones aren't - those are +available as calls back into the perf executable (see below). + +As an example, the following perf record command can be used to record +all sched_wakeup events in the system: + + # perf record -a -e sched:sched_wakeup + +Traces meant to be processed using a script should be recorded with +the above option: -a to enable system-wide collection. + +The format file for the sched_wakep event defines the following fields +(see /sys/kernel/debug/tracing/events/sched/sched_wakeup/format): + +---- + format: + field:unsigned short common_type; + field:unsigned char common_flags; + field:unsigned char common_preempt_count; + field:int common_pid; + + field:char comm[TASK_COMM_LEN]; + field:pid_t pid; + field:int prio; + field:int success; + field:int target_cpu; +---- + +The handler function for this event would be defined as: + +---- +def sched__sched_wakeup(event_name, context, common_cpu, common_secs, + common_nsecs, common_pid, common_comm, + comm, pid, prio, success, target_cpu): + pass +---- + +The handler function takes the form subsystem__event_name. + +The common_* arguments in the handler's argument list are the set of +arguments passed to all event handlers; some of the fields correspond +to the common_* fields in the format file, but some are synthesized, +and some of the common_* fields aren't common enough to to be passed +to every event as arguments but are available as library functions. + +Here's a brief description of each of the invariant event args: + + event_name the name of the event as text + context an opaque 'cookie' used in calls back into perf + common_cpu the cpu the event occurred on + common_secs the secs portion of the event timestamp + common_nsecs the nsecs portion of the event timestamp + common_pid the pid of the current task + common_comm the name of the current process + +All of the remaining fields in the event's format file have +counterparts as handler function arguments of the same name, as can be +seen in the example above. + +The above provides the basics needed to directly access every field of +every event in a trace, which covers 90% of what you need to know to +write a useful trace script. The sections below cover the rest. + +SCRIPT LAYOUT +------------- + +Every perf script Python script should start by setting up a Python +module search path and 'import'ing a few support modules (see module +descriptions below): + +---- + import os + import sys + + sys.path.append(os.environ['PERF_EXEC_PATH'] + \ + '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') + + from perf_trace_context import * + from Core import * +---- + +The rest of the script can contain handler functions and support +functions in any order. + +Aside from the event handler functions discussed above, every script +can implement a set of optional functions: + +*trace_begin*, if defined, is called before any event is processed and +gives scripts a chance to do setup tasks: + +---- +def trace_begin: + pass +---- + +*trace_end*, if defined, is called after all events have been + processed and gives scripts a chance to do end-of-script tasks, such + as display results: + +---- +def trace_end: + pass +---- + +*trace_unhandled*, if defined, is called after for any event that + doesn't have a handler explicitly defined for it. The standard set + of common arguments are passed into it: + +---- +def trace_unhandled(event_name, context, common_cpu, common_secs, + common_nsecs, common_pid, common_comm): + pass +---- + +The remaining sections provide descriptions of each of the available +built-in perf script Python modules and their associated functions. + +AVAILABLE MODULES AND FUNCTIONS +------------------------------- + +The following sections describe the functions and variables available +via the various perf script Python modules. To use the functions and +variables from the given module, add the corresponding 'from XXXX +import' line to your perf script script. + +Core.py Module +~~~~~~~~~~~~~~ + +These functions provide some essential functions to user scripts. + +The *flag_str* and *symbol_str* functions provide human-readable +strings for flag and symbolic fields. These correspond to the strings +and values parsed from the 'print fmt' fields of the event format +files: + + flag_str(event_name, field_name, field_value) - returns the string represention corresponding to field_value for the flag field field_name of event event_name + symbol_str(event_name, field_name, field_value) - returns the string represention corresponding to field_value for the symbolic field field_name of event event_name + +The *autodict* function returns a special kind of Python +dictionary that implements Perl's 'autovivifying' hashes in Python +i.e. with autovivifying hashes, you can assign nested hash values +without having to go to the trouble of creating intermediate levels if +they don't exist. + + autodict() - returns an autovivifying dictionary instance + + +perf_trace_context Module +~~~~~~~~~~~~~~~~~~~~~~~~~ + +Some of the 'common' fields in the event format file aren't all that +common, but need to be made accessible to user scripts nonetheless. + +perf_trace_context defines a set of functions that can be used to +access this data in the context of the current event. Each of these +functions expects a context variable, which is the same as the +context variable passed into every event handler as the second +argument. + + common_pc(context) - returns common_preempt count for the current event + common_flags(context) - returns common_flags for the current event + common_lock_depth(context) - returns common_lock_depth for the current event + +Util.py Module +~~~~~~~~~~~~~~ + +Various utility functions for use with perf script: + + nsecs(secs, nsecs) - returns total nsecs given secs/nsecs pair + nsecs_secs(nsecs) - returns whole secs portion given nsecs + nsecs_nsecs(nsecs) - returns nsecs remainder given nsecs + nsecs_str(nsecs) - returns printable string in the form secs.nsecs + avg(total, n) - returns average given a sum and a total number of values + +SEE ALSO +-------- +linkperf:perf-script[1] diff --git a/tools/perf/Documentation/perf-script.txt b/tools/perf/Documentation/perf-script.txt new file mode 100644 index 000000000..e9cbfcddf --- /dev/null +++ b/tools/perf/Documentation/perf-script.txt @@ -0,0 +1,209 @@ +perf-script(1) +============= + +NAME +---- +perf-script - Read perf.data (created by perf record) and display trace output + +SYNOPSIS +-------- +[verse] +'perf script' [] +'perf script' [] record