struct update_callback_data {
        int flags;
        int add_errors;
+
+       /* only needed for 2.0 transition preparation */
+       int warn_add_would_remove;
 };
 
 static int fix_unmerged_status(struct diff_filepair *p,
                return DIFF_STATUS_MODIFIED;
 }
 
+static void warn_add_would_remove(const char *path)
+{
+       warning(_("In Git 2.0, 'git add <pathspec>...' will also update the\n"
+                 "index for paths removed from the working tree that match\n"
+                 "the given pathspec. If you want to 'add' only changed\n"
+                 "or newly created paths, say 'git add --no-all <pathspec>...'"
+                 " instead.\n\n"
+                 "'%s' would be removed from the index without --no-all."),
+               path);
+}
+
 static void update_callback(struct diff_queue_struct *q,
                            struct diff_options *opt, void *cbdata)
 {
                        }
                        break;
                case DIFF_STATUS_DELETED:
+                       if (data->warn_add_would_remove) {
+                               warn_add_would_remove(path);
+                               data->warn_add_would_remove = 0;
+                       }
                        if (data->flags & ADD_CACHE_IGNORE_REMOVAL)
                                break;
                        if (!(data->flags & ADD_CACHE_PRETEND))
        }
 }
 
-int add_files_to_cache(const char *prefix, const char **pathspec, int flags)
+static void update_files_in_cache(const char *prefix, const char **pathspec,
+                                 struct update_callback_data *data)
 {
-       struct update_callback_data data;
        struct rev_info rev;
        init_revisions(&rev, prefix);
        setup_revisions(0, NULL, &rev, NULL);
        init_pathspec(&rev.prune_data, pathspec);
        rev.diffopt.output_format = DIFF_FORMAT_CALLBACK;
        rev.diffopt.format_callback = update_callback;
-       data.flags = flags;
-       data.add_errors = 0;
-       rev.diffopt.format_callback_data = &data;
+       rev.diffopt.format_callback_data = data;
        rev.max_count = 0; /* do not compare unmerged paths with stage #2 */
        run_diff_files(&rev, DIFF_RACY_IS_MODIFIED);
+}
+
+int add_files_to_cache(const char *prefix, const char **pathspec, int flags)
+{
+       struct update_callback_data data;
+
+       memset(&data, 0, sizeof(data));
+       data.flags = flags;
+       update_files_in_cache(prefix, pathspec, &data);
        return !!data.add_errors;
 }
 
                option_name, short_name);
 }
 
-static int directory_given(int argc, const char **argv)
-{
-       struct stat st;
-
-       while (argc--) {
-               if (!lstat(*argv, &st) && S_ISDIR(st.st_mode))
-                       return 1;
-               argv++;
-       }
-       return 0;
-}
-
 int cmd_add(int argc, const char **argv, const char *prefix)
 {
        int exit_status = 0;
        char *seen = NULL;
        const char *option_with_implicit_dot = NULL;
        const char *short_option_with_implicit_dot = NULL;
+       struct update_callback_data update_data;
 
        git_config(add_config, NULL);
 
 
        /*
         * Warn when "git add pathspec..." was given without "-u" or "-A"
-        * and pathspec... contains a directory name.
+        * and pathspec... covers a removed path.
         */
-       if (!take_worktree_changes && addremove_explicit < 0 &&
-           directory_given(argc, argv))
-               warning(_("In Git 2.0, 'git add <pathspec>...' will also update the\n"
-                         "index for paths removed from the working tree that match\n"
-                         "the given pathspec. If you want to 'add' only changed\n"
-                         "or newly created paths, say 'git add --no-all <pathspec>...'"
-                         " instead."));
+       memset(&update_data, 0, sizeof(update_data));
+       if (!take_worktree_changes && addremove_explicit < 0)
+               update_data.warn_add_would_remove = 1;
 
        if (!take_worktree_changes && addremove_explicit < 0 && argc)
                /*
 
        plug_bulk_checkin();
 
-       exit_status |= add_files_to_cache(prefix, pathspec, flags);
+       update_data.flags = flags;
+       update_files_in_cache(prefix, pathspec, &update_data);
 
+       exit_status |= !!update_data.add_errors;
        if (add_new_files)
                exit_status |= add_files(&dir, flags);