static int alloc_mad(struct ib_sa_query *query, gfp_t gfp_mask)
 {
+       struct rdma_ah_attr ah_attr;
        unsigned long flags;
 
        spin_lock_irqsave(&query->port->ah_lock, flags);
        query->sm_ah = query->port->sm_ah;
        spin_unlock_irqrestore(&query->port->ah_lock, flags);
 
+       /*
+        * Always check if sm_ah has valid dlid assigned,
+        * before querying for class port info
+        */
+       if ((rdma_query_ah(query->sm_ah->ah, &ah_attr) < 0) ||
+           !rdma_is_valid_unicast_lid(&ah_attr)) {
+               kref_put(&query->sm_ah->ref, free_sm_ah);
+               return -EAGAIN;
+       }
        query->mad_buf = ib_create_send_mad(query->port->agent, 1,
                                            query->sm_ah->pkey_index,
                                            0, IB_MGMT_SA_HDR, IB_MGMT_SA_DATA,
 
        return (be32_to_cpu(OPA_LID_PERMISSIVE) << (32 - nr_top_bits));
 }
 
+/* Check for a valid unicast LID for non-SM traffic types */
+static inline bool rdma_is_valid_unicast_lid(struct rdma_ah_attr *attr)
+{
+       if (attr->type == RDMA_AH_ATTR_TYPE_IB) {
+               if (!rdma_ah_get_dlid(attr) ||
+                   rdma_ah_get_dlid(attr) >=
+                   be32_to_cpu(IB_MULTICAST_LID_BASE))
+                       return false;
+       } else if (attr->type == RDMA_AH_ATTR_TYPE_OPA) {
+               if (!rdma_ah_get_dlid(attr) ||
+                   rdma_ah_get_dlid(attr) >=
+                   opa_get_mcast_base(OPA_MCAST_NR))
+                       return false;
+       }
+       return true;
+}
 #endif /* OPA_ADDR_H */