hw/block/nvme: trigger async event during injecting smart warning
During smart critical warning injection by setting property from QMP command, also try to trigger asynchronous event. Suggested by Keith, if a event has already been raised, there is no need to enqueue the duplicate event any more. Signed-off-by: zhenwei pi <pizhenwei@bytedance.com> [k.jensen: fix typo in commit message] Signed-off-by: Klaus Jensen <k.jensen@samsung.com>
This commit is contained in:
		| @@ -980,6 +980,35 @@ static void nvme_enqueue_event(NvmeCtrl *n, uint8_t event_type, | ||||
|     nvme_process_aers(n); | ||||
| } | ||||
|  | ||||
| static void nvme_smart_event(NvmeCtrl *n, uint8_t event) | ||||
| { | ||||
|     uint8_t aer_info; | ||||
|  | ||||
|     /* Ref SPEC <Asynchronous Event Information 0x2013 SMART / Health Status> */ | ||||
|     if (!(NVME_AEC_SMART(n->features.async_config) & event)) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     switch (event) { | ||||
|     case NVME_SMART_SPARE: | ||||
|         aer_info = NVME_AER_INFO_SMART_SPARE_THRESH; | ||||
|         break; | ||||
|     case NVME_SMART_TEMPERATURE: | ||||
|         aer_info = NVME_AER_INFO_SMART_TEMP_THRESH; | ||||
|         break; | ||||
|     case NVME_SMART_RELIABILITY: | ||||
|     case NVME_SMART_MEDIA_READ_ONLY: | ||||
|     case NVME_SMART_FAILED_VOLATILE_MEDIA: | ||||
|     case NVME_SMART_PMR_UNRELIABLE: | ||||
|         aer_info = NVME_AER_INFO_SMART_RELIABILITY; | ||||
|         break; | ||||
|     default: | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     nvme_enqueue_event(n, NVME_AER_TYPE_SMART, aer_info, NVME_LOG_SMART_INFO); | ||||
| } | ||||
|  | ||||
| static void nvme_clear_events(NvmeCtrl *n, uint8_t event_type) | ||||
| { | ||||
|     n->aer_mask &= ~(1 << event_type); | ||||
| @@ -3317,12 +3346,9 @@ static uint16_t nvme_set_feature(NvmeCtrl *n, NvmeRequest *req) | ||||
|             return NVME_INVALID_FIELD | NVME_DNR; | ||||
|         } | ||||
|  | ||||
|         if (((n->temperature >= n->features.temp_thresh_hi) || | ||||
|              (n->temperature <= n->features.temp_thresh_low)) && | ||||
|             NVME_AEC_SMART(n->features.async_config) & NVME_SMART_TEMPERATURE) { | ||||
|             nvme_enqueue_event(n, NVME_AER_TYPE_SMART, | ||||
|                                NVME_AER_INFO_SMART_TEMP_THRESH, | ||||
|                                NVME_LOG_SMART_INFO); | ||||
|         if ((n->temperature >= n->features.temp_thresh_hi) || | ||||
|             (n->temperature <= n->features.temp_thresh_low)) { | ||||
|             nvme_smart_event(n, NVME_AER_INFO_SMART_TEMP_THRESH); | ||||
|         } | ||||
|  | ||||
|         break; | ||||
| @@ -4446,7 +4472,7 @@ static void nvme_set_smart_warning(Object *obj, Visitor *v, const char *name, | ||||
|                                    void *opaque, Error **errp) | ||||
| { | ||||
|     NvmeCtrl *n = NVME(obj); | ||||
|     uint8_t value, cap = 0; | ||||
|     uint8_t value, old_value, cap = 0, index, event; | ||||
|  | ||||
|     if (!visit_type_uint8(v, name, &value, errp)) { | ||||
|         return; | ||||
| @@ -4464,7 +4490,15 @@ static void nvme_set_smart_warning(Object *obj, Visitor *v, const char *name, | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     old_value = n->smart_critical_warning; | ||||
|     n->smart_critical_warning = value; | ||||
|  | ||||
|     /* only inject new bits of smart critical warning */ | ||||
|     for (index = 0; index < NVME_SMART_WARN_MAX; index++) { | ||||
|         event = 1 << index; | ||||
|         if (value & ~old_value & event) | ||||
|             nvme_smart_event(n, event); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static const VMStateDescription nvme_vmstate = { | ||||
|   | ||||
| @@ -784,6 +784,7 @@ typedef struct QEMU_PACKED NvmeSmartLog { | ||||
|     uint8_t     reserved2[320]; | ||||
| } NvmeSmartLog; | ||||
|  | ||||
| #define NVME_SMART_WARN_MAX     6 | ||||
| enum NvmeSmartWarn { | ||||
|     NVME_SMART_SPARE                  = 1 << 0, | ||||
|     NVME_SMART_TEMPERATURE            = 1 << 1, | ||||
|   | ||||
		Reference in New Issue
	
	Block a user