struct fprobe;
 
 typedef int (*fprobe_entry_cb)(struct fprobe *fp, unsigned long entry_ip,
-                              unsigned long ret_ip, struct pt_regs *regs,
+                              unsigned long ret_ip, struct ftrace_regs *regs,
                               void *entry_data);
 
 typedef void (*fprobe_exit_cb)(struct fprobe *fp, unsigned long entry_ip,
 
 config FPROBE
        bool "Kernel Function Probe (fprobe)"
        depends on FUNCTION_TRACER
-       depends on DYNAMIC_FTRACE_WITH_REGS
+       depends on DYNAMIC_FTRACE_WITH_REGS || DYNAMIC_FTRACE_WITH_ARGS
        depends on HAVE_RETHOOK
        select RETHOOK
        default n
        select TRACING
        select PROBE_EVENTS
        select DYNAMIC_EVENTS
+       depends on DYNAMIC_FTRACE_WITH_REGS
        default y
        help
          This allows user to add tracing events on the function entry and
 
        void *data;
 };
 
-#ifdef CONFIG_FPROBE
+#if defined(CONFIG_FPROBE) && defined(CONFIG_DYNAMIC_FTRACE_WITH_REGS)
 struct bpf_kprobe_multi_link {
        struct bpf_link link;
        struct fprobe fp;
 
 static int
 kprobe_multi_link_handler(struct fprobe *fp, unsigned long fentry_ip,
-                         unsigned long ret_ip, struct pt_regs *regs,
+                         unsigned long ret_ip, struct ftrace_regs *fregs,
                          void *data)
 {
+       struct pt_regs *regs = ftrace_get_regs(fregs);
        struct bpf_kprobe_multi_link *link;
        int err;
 
+       if (!regs)
+               return 0;
+
        link = container_of(fp, struct bpf_kprobe_multi_link, fp);
        err = kprobe_multi_link_prog_run(link, get_entry_ip(fentry_ip), regs, false, data);
        return is_kprobe_session(link->link.prog) ? err : 0;
        kvfree(cookies);
        return err;
 }
-#else /* !CONFIG_FPROBE */
+#else /* !CONFIG_FPROBE || !CONFIG_DYNAMIC_FTRACE_WITH_REGS */
 int bpf_kprobe_multi_link_attach(const union bpf_attr *attr, struct bpf_prog *prog)
 {
        return -EOPNOTSUPP;
 
        }
 
        if (fp->entry_handler)
-               ret = fp->entry_handler(fp, ip, parent_ip, ftrace_get_regs(fregs), entry_data);
+               ret = fp->entry_handler(fp, ip, parent_ip, fregs, entry_data);
 
        /* If entry_handler returns !0, nmissed is not counted. */
        if (rh) {
                fp->ops.func = fprobe_kprobe_handler;
        else
                fp->ops.func = fprobe_handler;
+
        fp->ops.flags |= FTRACE_OPS_FL_SAVE_REGS;
 }
 
 
 
 /* function exit handler */
 static int trace_fprobe_entry_handler(struct fprobe *fp, unsigned long entry_ip,
-                               unsigned long ret_ip, struct pt_regs *regs,
+                               unsigned long ret_ip, struct ftrace_regs *fregs,
                                void *entry_data)
 {
        struct trace_fprobe *tf = container_of(fp, struct trace_fprobe, fp);
+       struct pt_regs *regs = ftrace_get_regs(fregs);
 
-       if (tf->tp.entry_arg)
+       if (regs && tf->tp.entry_arg)
                store_trace_entry_data(entry_data, &tf->tp, regs);
 
        return 0;
 #endif /* CONFIG_PERF_EVENTS */
 
 static int fentry_dispatcher(struct fprobe *fp, unsigned long entry_ip,
-                            unsigned long ret_ip, struct pt_regs *regs,
+                            unsigned long ret_ip, struct ftrace_regs *fregs,
                             void *entry_data)
 {
        struct trace_fprobe *tf = container_of(fp, struct trace_fprobe, fp);
+       struct pt_regs *regs = ftrace_get_regs(fregs);
        int ret = 0;
 
+       if (!regs)
+               return 0;
+
        if (trace_probe_test_flag(&tf->tp, TP_FLAG_TRACE))
                fentry_trace_func(tf, entry_ip, regs);
 #ifdef CONFIG_PERF_EVENTS
 
 
 static notrace int fp_entry_handler(struct fprobe *fp, unsigned long ip,
                                    unsigned long ret_ip,
-                                   struct pt_regs *regs, void *data)
+                                   struct ftrace_regs *fregs, void *data)
 {
        KUNIT_EXPECT_FALSE(current_test, preemptible());
        /* This can be called on the fprobe_selftest_target and the fprobe_selftest_target2 */
 
 static notrace int nest_entry_handler(struct fprobe *fp, unsigned long ip,
                                      unsigned long ret_ip,
-                                     struct pt_regs *regs, void *data)
+                                     struct ftrace_regs *fregs, void *data)
 {
        KUNIT_EXPECT_FALSE(current_test, preemptible());
        return 0;
 
 
 static int sample_entry_handler(struct fprobe *fp, unsigned long ip,
                                unsigned long ret_ip,
-                               struct pt_regs *regs, void *data)
+                               struct ftrace_regs *fregs, void *data)
 {
        if (use_trace)
                /*