j->ignore_order = j->ignore_order || other->ignore_order;
}
-Job* job_install(Job *j, JobMode mode) {
+Job* job_install(Job *j, bool refuse_late_merge) {
Job **pj;
Job *uj;
+ assert(j);
assert(!j->installed);
assert(j->type < _JOB_TYPE_MAX_IN_TRANSACTION);
assert(j->state == JOB_WAITING);
/* not conflicting, i.e. mergeable */
if (uj->state == JOB_WAITING ||
- (job_type_allows_late_merge(j->type) && mode != JOB_RESTART_DEPENDENCIES && job_type_is_superset(uj->type, j->type))) {
+ (!refuse_late_merge && job_type_allows_late_merge(j->type) && job_type_is_superset(uj->type, j->type))) {
job_merge_into_installed(uj, j);
log_unit_debug(uj->unit,
"Merged %s/%s into installed job %s/%s as %"PRIu32,
Job* job_new_raw(Unit *unit);
void job_unlink(Job *job);
Job* job_free(Job *job);
-Job* job_install(Job *j, JobMode mode);
+Job* job_install(Job *j, bool refuse_late_merge);
int job_install_deserialized(Job *j);
void job_uninstall(Job *j);
void job_dump(Job *j, FILE *f, const char *prefix);
/* Clean the job dependencies */
transaction_unlink_job(tr, j, false);
- installed_job = job_install(j, mode);
+ /* When RestartMode=direct is used, the service being restarted don't enter the inactive/failed
+ * state, i.e. unit_process_job -> job_finish_and_invalidate is never called, and the previous
+ * job might still be running (especially for Type=oneshot services). We need to refuse
+ * late merge and re-enqueue the anchor job. */
+ installed_job = job_install(j,
+ /* refuse_late_merge = */ mode == JOB_RESTART_DEPENDENCIES && j == tr->anchor_job);
if (installed_job != j) {
/* j has been merged into a previously installed job */
if (tr->anchor_job == j)
tr->anchor_job = installed_job;
+
hashmap_remove_value(m->jobs, UINT32_TO_PTR(j->id), j);
- job_free(j);
- j = installed_job;
+ free_and_replace_full(j, installed_job, job_free);
}
job_add_to_run_queue(j);