* Collect all the non-memory information about the process for the
         * notes.  This also sets up the file header.
         */
-       if (!fill_note_info(&elf, e_phnum, &info, cprm))
+       if (!fill_note_info(&elf, e_phnum, &info, cprm)) {
+               coredump_report_failure("Error collecting note info");
                goto end_coredump;
+       }
 
        has_dumped = 1;
 
                sz += elf_coredump_extra_notes_size();
 
                phdr4note = kmalloc(sizeof(*phdr4note), GFP_KERNEL);
-               if (!phdr4note)
+               if (!phdr4note) {
+                       coredump_report_failure("Error allocating program headers note entry");
                        goto end_coredump;
+               }
 
                fill_elf_note_phdr(phdr4note, sz, offset);
                offset += sz;
 
        if (e_phnum == PN_XNUM) {
                shdr4extnum = kmalloc(sizeof(*shdr4extnum), GFP_KERNEL);
-               if (!shdr4extnum)
+               if (!shdr4extnum) {
+                       coredump_report_failure("Error allocating extra program headers");
                        goto end_coredump;
+               }
                fill_extnum_info(&elf, shdr4extnum, e_shoff, segs);
        }
 
        offset = dataoff;
 
-       if (!dump_emit(cprm, &elf, sizeof(elf)))
+       if (!dump_emit(cprm, &elf, sizeof(elf))) {
+               coredump_report_failure("Error emitting the ELF headers");
                goto end_coredump;
+       }
 
-       if (!dump_emit(cprm, phdr4note, sizeof(*phdr4note)))
+       if (!dump_emit(cprm, phdr4note, sizeof(*phdr4note))) {
+               coredump_report_failure("Error emitting the program header for notes");
                goto end_coredump;
+       }
 
        /* Write program headers for segments dump */
        for (i = 0; i < cprm->vma_count; i++) {
                        phdr.p_flags |= PF_X;
                phdr.p_align = ELF_EXEC_PAGESIZE;
 
-               if (!dump_emit(cprm, &phdr, sizeof(phdr)))
+               if (!dump_emit(cprm, &phdr, sizeof(phdr))) {
+                       coredump_report_failure("Error emitting program headers");
                        goto end_coredump;
+               }
        }
 
-       if (!elf_core_write_extra_phdrs(cprm, offset))
+       if (!elf_core_write_extra_phdrs(cprm, offset)) {
+               coredump_report_failure("Error writing out extra program headers");
                goto end_coredump;
+       }
 
        /* write out the notes section */
-       if (!write_note_info(&info, cprm))
+       if (!write_note_info(&info, cprm)) {
+               coredump_report_failure("Error writing out notes");
                goto end_coredump;
+       }
 
        /* For cell spufs */
-       if (elf_coredump_extra_notes_write(cprm))
+       if (elf_coredump_extra_notes_write(cprm)) {
+               coredump_report_failure("Error writing out extra notes");
                goto end_coredump;
+       }
 
        /* Align to page */
        dump_skip_to(cprm, dataoff);
        for (i = 0; i < cprm->vma_count; i++) {
                struct core_vma_metadata *meta = cprm->vma_meta + i;
 
-               if (!dump_user_range(cprm, meta->start, meta->dump_size))
+               if (!dump_user_range(cprm, meta->start, meta->dump_size)) {
+                       coredump_report_failure("Error writing out the process memory");
                        goto end_coredump;
+               }
        }
 
-       if (!elf_core_write_extra_data(cprm))
+       if (!elf_core_write_extra_data(cprm)) {
+               coredump_report_failure("Error writing out extra data");
                goto end_coredump;
+       }
 
        if (e_phnum == PN_XNUM) {
-               if (!dump_emit(cprm, shdr4extnum, sizeof(*shdr4extnum)))
+               if (!dump_emit(cprm, shdr4extnum, sizeof(*shdr4extnum))) {
+                       coredump_report_failure("Error emitting extra program headers");
                        goto end_coredump;
+               }
        }
 
 end_coredump:
 
         * but then we need to teach dump_write() to restart and clear
         * TIF_SIGPENDING.
         */
-       return fatal_signal_pending(current) || freezing(current);
+       if (fatal_signal_pending(current)) {
+               coredump_report_failure("interrupted: fatal signal pending");
+               return true;
+       }
+
+       if (freezing(current)) {
+               coredump_report_failure("interrupted: freezing");
+               return true;
+       }
+
+       return false;
 }
 
 static void wait_for_dump_helpers(struct file *file)
        return err;
 }
 
-void do_coredump(const kernel_siginfo_t *siginfo)
+int do_coredump(const kernel_siginfo_t *siginfo)
 {
        struct core_state core_state;
        struct core_name cn;
        struct linux_binfmt * binfmt;
        const struct cred *old_cred;
        struct cred *cred;
-       int retval = 0;
+       int retval;
        int ispipe;
        size_t *argv = NULL;
        int argc = 0;
        audit_core_dumps(siginfo->si_signo);
 
        binfmt = mm->binfmt;
-       if (!binfmt || !binfmt->core_dump)
+       if (!binfmt || !binfmt->core_dump) {
+               retval = -ENOEXEC;
                goto fail;
-       if (!__get_dumpable(cprm.mm_flags))
+       }
+       if (!__get_dumpable(cprm.mm_flags)) {
+               retval = -EACCES;
                goto fail;
+       }
 
        cred = prepare_creds();
-       if (!cred)
+       if (!cred) {
+               retval = -EPERM;
                goto fail;
+       }
        /*
         * We cannot trust fsuid as being the "true" uid of the process
         * nor do we know its entire history. We only know it was tainted
 
                if (ispipe < 0) {
                        coredump_report_failure("format_corename failed, aborting core");
+                       retval = ispipe;
                        goto fail_unlock;
                }
 
                         * core_pattern process dies.
                         */
                        coredump_report_failure("RLIMIT_CORE is set to 1, aborting core");
+                       retval = -EPERM;
                        goto fail_unlock;
                }
                cprm.limit = RLIM_INFINITY;
                dump_count = atomic_inc_return(&core_dump_count);
                if (core_pipe_limit && (core_pipe_limit < dump_count)) {
                        coredump_report_failure("over core_pipe_limit, skipping core dump");
+                       retval = -E2BIG;
                        goto fail_dropcount;
                }
 
                                            GFP_KERNEL);
                if (!helper_argv) {
                        coredump_report_failure("%s failed to allocate memory", __func__);
+                       retval = -ENOMEM;
                        goto fail_dropcount;
                }
                for (argi = 0; argi < argc; argi++)
                int open_flags = O_CREAT | O_WRONLY | O_NOFOLLOW |
                                 O_LARGEFILE | O_EXCL;
 
-               if (cprm.limit < binfmt->min_coredump)
+               if (cprm.limit < binfmt->min_coredump) {
+                       coredump_report_failure("over coredump resource limit, skipping core dump");
+                       retval = -E2BIG;
                        goto fail_unlock;
+               }
 
                if (need_suid_safe && cn.corename[0] != '/') {
                        coredump_report_failure(
                                "this process can only dump core to a fully qualified path, skipping core dump");
+                       retval = -EPERM;
                        goto fail_unlock;
                }
 
                } else {
                        cprm.file = filp_open(cn.corename, open_flags, 0600);
                }
-               if (IS_ERR(cprm.file))
+               if (IS_ERR(cprm.file)) {
+                       retval = PTR_ERR(cprm.file);
                        goto fail_unlock;
+               }
 
                inode = file_inode(cprm.file);
-               if (inode->i_nlink > 1)
+               if (inode->i_nlink > 1) {
+                       retval = -EMLINK;
                        goto close_fail;
-               if (d_unhashed(cprm.file->f_path.dentry))
+               }
+               if (d_unhashed(cprm.file->f_path.dentry)) {
+                       retval = -EEXIST;
                        goto close_fail;
+               }
                /*
                 * AK: actually i see no reason to not allow this for named
                 * pipes etc, but keep the previous behaviour for now.
                 */
-               if (!S_ISREG(inode->i_mode))
+               if (!S_ISREG(inode->i_mode)) {
+                       retval = -EISDIR;
                        goto close_fail;
+               }
                /*
                 * Don't dump core if the filesystem changed owner or mode
                 * of the file during file creation. This is an issue when
                                    current_fsuid())) {
                        coredump_report_failure("Core dump to %s aborted: "
                                "cannot preserve file owner", cn.corename);
+                       retval = -EPERM;
                        goto close_fail;
                }
                if ((inode->i_mode & 0677) != 0600) {
                        coredump_report_failure("Core dump to %s aborted: "
                                "cannot preserve file permissions", cn.corename);
+                       retval = -EPERM;
                        goto close_fail;
                }
-               if (!(cprm.file->f_mode & FMODE_CAN_WRITE))
+               if (!(cprm.file->f_mode & FMODE_CAN_WRITE)) {
+                       retval = -EACCES;
                        goto close_fail;
-               if (do_truncate(idmap, cprm.file->f_path.dentry,
-                               0, 0, cprm.file))
+               }
+               retval = do_truncate(idmap, cprm.file->f_path.dentry,
+                               0, 0, cprm.file);
+               if (retval)
                        goto close_fail;
        }
 
                 */
                if (!cprm.file) {
                        coredump_report_failure("Core dump to |%s disabled", cn.corename);
+                       retval = -EPERM;
                        goto close_fail;
                }
-               if (!dump_vma_snapshot(&cprm))
+               if (!dump_vma_snapshot(&cprm)) {
+                       coredump_report_failure("Can't get VMA snapshot for core dump |%s",
+                               cn.corename);
+                       retval = -EACCES;
                        goto close_fail;
+               }
 
                file_start_write(cprm.file);
                core_dumped = binfmt->core_dump(&cprm);
                }
                file_end_write(cprm.file);
                free_vma_snapshot(&cprm);
+       } else {
+               coredump_report_failure("Core dump to %s%s has been interrupted",
+                       ispipe ? "|" : "", cn.corename);
+               retval = -EAGAIN;
+               goto fail;
        }
+       coredump_report(
+               "written to %s%s: VMAs: %d, size %zu; core: %lld bytes, pos %lld",
+               ispipe ? "|" : "", cn.corename,
+               cprm.vma_count, cprm.vma_data_size, cprm.written, cprm.pos);
        if (ispipe && core_pipe_limit)
                wait_for_dump_helpers(cprm.file);
+
+       retval = 0;
+
 close_fail:
        if (cprm.file)
                filp_close(cprm.file, NULL);
 fail_creds:
        put_cred(cred);
 fail:
-       return;
+       return retval;
 }
 
 /*
        if (dump_interrupted())
                return 0;
        n = __kernel_write(file, addr, nr, &pos);
-       if (n != nr)
+       if (n != nr) {
+               if (n < 0)
+                       coredump_report_failure("failed when writing out, error %zd", n);
+               else
+                       coredump_report_failure(
+                               "partially written out, only %zd(of %d) bytes written",
+                               n, nr);
+
                return 0;
+       }
        file->f_pos = pos;
        cprm->written += n;
        cprm->pos += n;
        static char zeroes[PAGE_SIZE];
        struct file *file = cprm->file;
        if (file->f_mode & FMODE_LSEEK) {
-               if (dump_interrupted() ||
-                   vfs_llseek(file, nr, SEEK_CUR) < 0)
+               int ret;
+
+               if (dump_interrupted())
                        return 0;
+
+               ret = vfs_llseek(file, nr, SEEK_CUR);
+               if (ret < 0) {
+                       coredump_report_failure("failed when seeking, error %d", ret);
+                       return 0;
+               }
                cprm->pos += nr;
                return 1;
        } else {
 
                current->flags |= PF_SIGNALED;
 
                if (sig_kernel_coredump(signr)) {
+                       int ret;
+
                        if (print_fatal_signals)
                                print_fatal_signal(signr);
                        proc_coredump_connector(current);
                         * first and our do_group_exit call below will use
                         * that value and ignore the one we pass it.
                         */
-                       do_coredump(&ksig->info);
+                       ret = do_coredump(&ksig->info);
+                       if (ret)
+                               coredump_report_failure("coredump has not been created, error %d",
+                                       ret);
+                       else if (!IS_ENABLED(CONFIG_COREDUMP)) {
+                               /*
+                                * Coredumps are not available, can't fail collecting
+                                * the coredump.
+                                *
+                                * Leave a note though that the coredump is going to be
+                                * not created. This is not an error or a warning as disabling
+                                * support in the kernel for coredumps isn't commonplace, and
+                                * the user must've built the kernel with the custom config so
+                                * let them know all works as desired.
+                                */
+                               coredump_report("no coredump collected as "
+                                       "that is disabled in the kernel configuration");
+                       }
                }
 
                /*