int mem_size;
        u64 msize_max_value;
        int ref_obj_id;
+       int map_uid;
        int func_id;
        struct btf *btf;
        u32 btf_id;
                if (map->inner_map_meta) {
                        reg->type = CONST_PTR_TO_MAP;
                        reg->map_ptr = map->inner_map_meta;
+                       /* transfer reg's id which is unique for every map_lookup_elem
+                        * as UID of the inner map.
+                        */
+                       reg->map_uid = reg->id;
                } else if (map->map_type == BPF_MAP_TYPE_XSKMAP) {
                        reg->type = PTR_TO_XDP_SOCK;
                } else if (map->map_type == BPF_MAP_TYPE_SOCKMAP ||
                verbose(env, "verifier bug. Two map pointers in a timer helper\n");
                return -EFAULT;
        }
+       meta->map_uid = reg->map_uid;
        meta->map_ptr = map;
        return 0;
 }
 
        if (arg_type == ARG_CONST_MAP_PTR) {
                /* bpf_map_xxx(map_ptr) call: remember that map_ptr */
-               if (meta->map_ptr && meta->map_ptr != reg->map_ptr) {
-                       verbose(env, "Map pointer doesn't match bpf_timer.\n");
-                       return -EINVAL;
+               if (meta->map_ptr) {
+                       /* Use map_uid (which is unique id of inner map) to reject:
+                        * inner_map1 = bpf_map_lookup_elem(outer_map, key1)
+                        * inner_map2 = bpf_map_lookup_elem(outer_map, key2)
+                        * if (inner_map1 && inner_map2) {
+                        *     timer = bpf_map_lookup_elem(inner_map1);
+                        *     if (timer)
+                        *         // mismatch would have been allowed
+                        *         bpf_timer_init(timer, inner_map2);
+                        * }
+                        *
+                        * Comparing map_ptr is enough to distinguish normal and outer maps.
+                        */
+                       if (meta->map_ptr != reg->map_ptr ||
+                           meta->map_uid != reg->map_uid) {
+                               verbose(env,
+                                       "timer pointer in R1 map_uid=%d doesn't match map pointer in R2 map_uid=%d\n",
+                                       meta->map_uid, reg->map_uid);
+                               return -EINVAL;
+                       }
                }
                meta->map_ptr = reg->map_ptr;
+               meta->map_uid = reg->map_uid;
        } else if (arg_type == ARG_PTR_TO_MAP_KEY) {
                /* bpf_map_xxx(..., map_ptr, ..., key) call:
                 * check that [key, key + map->key_size) are within
                        return -EINVAL;
                }
                regs[BPF_REG_0].map_ptr = meta.map_ptr;
+               regs[BPF_REG_0].map_uid = meta.map_uid;
                if (fn->ret_type == RET_PTR_TO_MAP_VALUE) {
                        regs[BPF_REG_0].type = PTR_TO_MAP_VALUE;
                        if (map_value_has_spin_lock(meta.map_ptr))