struct nvmet_ctrl *ctrl = req->sq->ctrl;
        struct nvmet_subsys *subsys = ctrl->subsys;
        struct nvme_id_ctrl *id;
-       u32 cmd_capsule_size;
+       u32 cmd_capsule_size, ctratt;
        u16 status = 0;
 
        if (!subsys->subsys_discovered) {
 
        /* XXX: figure out what to do about RTD3R/RTD3 */
        id->oaes = cpu_to_le32(NVMET_AEN_CFG_OPTIONAL);
-       id->ctratt = cpu_to_le32(NVME_CTRL_ATTR_HID_128_BIT |
-               NVME_CTRL_ATTR_TBKAS);
+       ctratt = NVME_CTRL_ATTR_HID_128_BIT | NVME_CTRL_ATTR_TBKAS;
+       if (nvmet_is_pci_ctrl(ctrl))
+               ctratt |= NVME_CTRL_ATTR_RHII;
+       id->ctratt = cpu_to_le32(ctratt);
 
        id->oacs = 0;
 
        return 0;
 }
 
+static u16 nvmet_set_feat_host_id(struct nvmet_req *req)
+{
+       struct nvmet_ctrl *ctrl = req->sq->ctrl;
+
+       if (!nvmet_is_pci_ctrl(ctrl))
+               return NVME_SC_CMD_SEQ_ERROR | NVME_STATUS_DNR;
+
+       /*
+        * The NVMe base specifications v2.1 recommends supporting 128-bits host
+        * IDs (section 5.1.25.1.28.1). However, that same section also says
+        * that "The controller may support a 64-bit Host Identifier and/or an
+        * extended 128-bit Host Identifier". So simplify this support and do
+        * not support 64-bits host IDs to avoid needing to check that all
+        * controllers associated with the same subsystem all use the same host
+        * ID size.
+        */
+       if (!(req->cmd->common.cdw11 & cpu_to_le32(1 << 0))) {
+               req->error_loc = offsetof(struct nvme_common_command, cdw11);
+               return NVME_SC_INVALID_FIELD | NVME_STATUS_DNR;
+       }
+
+       return nvmet_copy_from_sgl(req, 0, &req->sq->ctrl->hostid,
+                                  sizeof(req->sq->ctrl->hostid));
+}
+
 void nvmet_execute_set_features(struct nvmet_req *req)
 {
        struct nvmet_subsys *subsys = nvmet_req_subsys(req);
                status = nvmet_set_feat_async_event(req, NVMET_AEN_CFG_ALL);
                break;
        case NVME_FEAT_HOST_ID:
-               status = NVME_SC_CMD_SEQ_ERROR | NVME_STATUS_DNR;
+               status = nvmet_set_feat_host_id(req);
                break;
        case NVME_FEAT_WRITE_PROTECT:
                status = nvmet_set_feat_write_protect(req);