return ast_sorcery_generic_alloc(sizeof(struct ast_sip_publication_resource), publication_resource_destroy);
 }
 
-static void sub_tree_transport_cb(void *data) {
+static int sub_tree_subscription_terminate_cb(void *data)
+{
        struct sip_subscription_tree *sub_tree = data;
 
-       ast_debug(3, "Transport destroyed.  Removing subscription '%s->%s'  prune on restart: %d\n",
+       if (!sub_tree->evsub) {
+               /* Something else already terminated the subscription. */
+               ao2_ref(sub_tree, -1);
+               return 0;
+       }
+
+       ast_debug(3, "Transport destroyed.  Removing subscription '%s->%s'  prune on boot: %d\n",
                sub_tree->persistence->endpoint, sub_tree->root->resource,
                sub_tree->persistence->prune_on_boot);
 
        sub_tree->state = SIP_SUB_TREE_TERMINATE_IN_PROGRESS;
        pjsip_evsub_terminate(sub_tree->evsub, PJ_TRUE);
+
+       ao2_ref(sub_tree, -1);
+       return 0;
+}
+
+/*!
+ * \internal
+ * \brief The reliable transport we used as a subscription contact has shutdown.
+ *
+ * \param data What subscription needs to be terminated.
+ *
+ * \note Normally executed by the pjsip monitor thread.
+ *
+ * \return Nothing
+ */
+static void sub_tree_transport_cb(void *data)
+{
+       struct sip_subscription_tree *sub_tree = data;
+
+       /*
+        * Push off the subscription termination to the serializer to
+        * avoid deadlock.  Another thread could be trying to send a
+        * message on the subscription that can deadlock with this
+        * thread.
+        */
+       ao2_ref(sub_tree, +1);
+       if (ast_sip_push_task(sub_tree->serializer, sub_tree_subscription_terminate_cb,
+               sub_tree)) {
+               ao2_ref(sub_tree, -1);
+       }
 }
 
 /*! \brief Destructor for subscription persistence */
                return;
        }
 
-       ast_debug(3, "Updating persistence for '%s->%s'  prune on restart: %s\n",
+       ast_debug(3, "Updating persistence for '%s->%s'  prune on boot: %s\n",
                sub_tree->persistence->endpoint, sub_tree->root->resource,
                sub_tree->persistence->prune_on_boot ? "yes" : "no");
 
                                                        sub_tree->endpoint, rdata);
 
                                        if (sub_tree->persistence->prune_on_boot) {
-                                               ast_debug(3, "adding transport monitor on %s for '%s->%s'  prune on restart: %d\n",
+                                               ast_debug(3, "adding transport monitor on %s for '%s->%s'  prune on boot: %d\n",
                                                        rdata->tp_info.transport->obj_name,
                                                        sub_tree->persistence->endpoint, sub_tree->root->resource,
                                                        sub_tree->persistence->prune_on_boot);
 
                && strcmp(ma->contact_name, mb->contact_name) == 0;
 }
 
-static void register_contact_transport_shutdown_cb(void *data)
+static int register_contact_transport_remove_cb(void *data)
 {
        struct contact_transport_monitor *monitor = data;
        struct ast_sip_contact *contact;
 
        aor = ast_sip_location_retrieve_aor(monitor->aor_name);
        if (!aor) {
-               return;
+               ao2_ref(monitor, -1);
+               return 0;
        }
 
        ao2_lock(aor);
        }
        ao2_unlock(aor);
        ao2_ref(aor, -1);
+
+       ao2_ref(monitor, -1);
+       return 0;
+}
+
+/*!
+ * \internal
+ * \brief The reliable transport we registered as a contact has shutdown.
+ *
+ * \param data What contact needs to be removed.
+ *
+ * \note Normally executed by the pjsip monitor thread.
+ *
+ * \return Nothing
+ */
+static void register_contact_transport_shutdown_cb(void *data)
+{
+       struct contact_transport_monitor *monitor = data;
+
+       /*
+        * Push off to a default serializer.  This is in case sorcery
+        * does database accesses for contacts.  Database accesses may
+        * not be on this machine.  We don't want to tie up the pjsip
+        * monitor thread with potentially long access times.
+        */
+       ao2_ref(monitor, +1);
+       if (ast_sip_push_task(NULL, register_contact_transport_remove_cb, monitor)) {
+               ao2_ref(monitor, -1);
+       }
 }
 
 AST_VECTOR(excess_contact_vector, struct ast_sip_contact *);