return retval;
 }
 
+static void wacom_wac_queue_insert(struct hid_device *hdev,
+                                  struct kfifo_rec_ptr_2 *fifo,
+                                  u8 *raw_data, int size)
+{
+       bool warned = false;
+
+       while (kfifo_avail(fifo) < size) {
+               if (!warned)
+                       hid_warn(hdev, "%s: kfifo has filled, starting to drop events\n", __func__);
+               warned = true;
+
+               kfifo_skip(fifo);
+       }
+
+       kfifo_in(fifo, raw_data, size);
+}
+
+static void wacom_wac_queue_flush(struct hid_device *hdev,
+                                 struct kfifo_rec_ptr_2 *fifo)
+{
+       while (!kfifo_is_empty(fifo)) {
+               u8 buf[WACOM_PKGLEN_MAX];
+               int size;
+               int err;
+
+               size = kfifo_out(fifo, buf, sizeof(buf));
+               err = hid_report_raw_event(hdev, HID_INPUT_REPORT, buf, size, false);
+               if (err) {
+                       hid_warn(hdev, "%s: unable to flush event due to error %d\n",
+                                __func__, err);
+               }
+       }
+}
+
+static int wacom_wac_pen_serial_enforce(struct hid_device *hdev,
+               struct hid_report *report, u8 *raw_data, int size)
+{
+       struct wacom *wacom = hid_get_drvdata(hdev);
+       struct wacom_wac *wacom_wac = &wacom->wacom_wac;
+       struct wacom_features *features = &wacom_wac->features;
+       bool flush = false;
+       bool insert = false;
+       int i, j;
+
+       if (wacom_wac->serial[0] || !(features->quirks & WACOM_QUIRK_TOOLSERIAL))
+               return 0;
+
+       /* Queue events which have invalid tool type or serial number */
+       for (i = 0; i < report->maxfield; i++) {
+               for (j = 0; j < report->field[i]->maxusage; j++) {
+                       struct hid_field *field = report->field[i];
+                       struct hid_usage *usage = &field->usage[j];
+                       unsigned int equivalent_usage = wacom_equivalent_usage(usage->hid);
+                       unsigned int offset;
+                       unsigned int size;
+                       unsigned int value;
+
+                       if (equivalent_usage != HID_DG_INRANGE &&
+                           equivalent_usage != HID_DG_TOOLSERIALNUMBER &&
+                           equivalent_usage != WACOM_HID_WD_SERIALHI &&
+                           equivalent_usage != WACOM_HID_WD_TOOLTYPE)
+                               continue;
+
+                       offset = field->report_offset;
+                       size = field->report_size;
+                       value = hid_field_extract(hdev, raw_data+1, offset + j * size, size);
+
+                       /* If we go out of range, we need to flush the queue ASAP */
+                       if (equivalent_usage == HID_DG_INRANGE)
+                               value = !value;
+
+                       if (value) {
+                               flush = true;
+                               switch (equivalent_usage) {
+                               case HID_DG_TOOLSERIALNUMBER:
+                                       wacom_wac->serial[0] = value;
+                                       break;
+
+                               case WACOM_HID_WD_SERIALHI:
+                                       wacom_wac->serial[0] |= ((__u64)value) << 32;
+                                       break;
+
+                               case WACOM_HID_WD_TOOLTYPE:
+                                       wacom_wac->id[0] = value;
+                                       break;
+                               }
+                       }
+                       else {
+                               insert = true;
+                       }
+               }
+       }
+
+       if (flush)
+               wacom_wac_queue_flush(hdev, &wacom_wac->pen_fifo);
+       else if (insert)
+               wacom_wac_queue_insert(hdev, &wacom_wac->pen_fifo, raw_data, size);
+
+       return insert && !flush;
+}
+
 static int wacom_raw_event(struct hid_device *hdev, struct hid_report *report,
                u8 *raw_data, int size)
 {
        if (size > WACOM_PKGLEN_MAX)
                return 1;
 
+       if (wacom_wac_pen_serial_enforce(hdev, report, raw_data, size))
+               return -1;
+
        memcpy(wacom->wacom_wac.data, raw_data, size);
 
        wacom_wac_irq(&wacom->wacom_wac, size);
                goto fail;
        }
 
+       error = kfifo_alloc(&wacom_wac->pen_fifo, WACOM_PKGLEN_MAX, GFP_KERNEL);
+       if (error)
+               goto fail;
+
        wacom_wac->hid_data.inputmode = -1;
        wacom_wac->mode_report = -1;
 
        if (wacom->wacom_wac.features.type != REMOTE)
                wacom_release_resources(wacom);
 
+       kfifo_free(&wacom_wac->pen_fifo);
+
        hid_set_drvdata(hdev, NULL);
 }