| // SPDX-License-Identifier: GPL-2.0 | 
 | /* | 
 |  * Apple Z2 touchscreen driver | 
 |  * | 
 |  * Copyright (C) The Asahi Linux Contributors | 
 |  */ | 
 |  | 
 | #include <linux/delay.h> | 
 | #include <linux/firmware.h> | 
 | #include <linux/input.h> | 
 | #include <linux/input/mt.h> | 
 | #include <linux/input/touchscreen.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/module.h> | 
 | #include <linux/of.h> | 
 | #include <linux/spi/spi.h> | 
 | #include <linux/unaligned.h> | 
 |  | 
 | #define APPLE_Z2_NUM_FINGERS_OFFSET      16 | 
 | #define APPLE_Z2_FINGERS_OFFSET          24 | 
 | #define APPLE_Z2_TOUCH_STARTED           3 | 
 | #define APPLE_Z2_TOUCH_MOVED             4 | 
 | #define APPLE_Z2_CMD_READ_INTERRUPT_DATA 0xEB | 
 | #define APPLE_Z2_HBPP_CMD_BLOB           0x3001 | 
 | #define APPLE_Z2_FW_MAGIC                0x5746325A | 
 | #define LOAD_COMMAND_INIT_PAYLOAD        0 | 
 | #define LOAD_COMMAND_SEND_BLOB           1 | 
 | #define LOAD_COMMAND_SEND_CALIBRATION    2 | 
 | #define CAL_PROP_NAME                    "apple,z2-cal-blob" | 
 |  | 
 | struct apple_z2 { | 
 | 	struct spi_device *spidev; | 
 | 	struct gpio_desc *reset_gpio; | 
 | 	struct input_dev *input_dev; | 
 | 	struct completion boot_irq; | 
 | 	bool booted; | 
 | 	int index_parity; | 
 | 	struct touchscreen_properties props; | 
 | 	const char *fw_name; | 
 | 	u8 *tx_buf; | 
 | 	u8 *rx_buf; | 
 | }; | 
 |  | 
 | struct apple_z2_finger { | 
 | 	u8 finger; | 
 | 	u8 state; | 
 | 	__le16 unknown2; | 
 | 	__le16 abs_x; | 
 | 	__le16 abs_y; | 
 | 	__le16 rel_x; | 
 | 	__le16 rel_y; | 
 | 	__le16 tool_major; | 
 | 	__le16 tool_minor; | 
 | 	__le16 orientation; | 
 | 	__le16 touch_major; | 
 | 	__le16 touch_minor; | 
 | 	__le16 unused[2]; | 
 | 	__le16 pressure; | 
 | 	__le16 multi; | 
 | } __packed; | 
 |  | 
 | struct apple_z2_hbpp_blob_hdr { | 
 | 	__le16 cmd; | 
 | 	__le16 len; | 
 | 	__le32 addr; | 
 | 	__le16 checksum; | 
 | }; | 
 |  | 
 | struct apple_z2_fw_hdr { | 
 | 	__le32 magic; | 
 | 	__le32 version; | 
 | }; | 
 |  | 
 | struct apple_z2_read_interrupt_cmd { | 
 | 	u8 cmd; | 
 | 	u8 counter; | 
 | 	u8 unused[12]; | 
 | 	__le16 checksum; | 
 | }; | 
 |  | 
 | static void apple_z2_parse_touches(struct apple_z2 *z2, | 
 | 				   const u8 *msg, size_t msg_len) | 
 | { | 
 | 	int i; | 
 | 	int nfingers; | 
 | 	int slot; | 
 | 	int slot_valid; | 
 | 	struct apple_z2_finger *fingers; | 
 |  | 
 | 	if (msg_len <= APPLE_Z2_NUM_FINGERS_OFFSET) | 
 | 		return; | 
 | 	nfingers = msg[APPLE_Z2_NUM_FINGERS_OFFSET]; | 
 | 	fingers = (struct apple_z2_finger *)(msg + APPLE_Z2_FINGERS_OFFSET); | 
 | 	for (i = 0; i < nfingers; i++) { | 
 | 		slot = input_mt_get_slot_by_key(z2->input_dev, fingers[i].finger); | 
 | 		if (slot < 0) { | 
 | 			dev_warn(&z2->spidev->dev, "unable to get slot for finger\n"); | 
 | 			continue; | 
 | 		} | 
 | 		slot_valid = fingers[i].state == APPLE_Z2_TOUCH_STARTED || | 
 | 			     fingers[i].state == APPLE_Z2_TOUCH_MOVED; | 
 | 		input_mt_slot(z2->input_dev, slot); | 
 | 		if (!input_mt_report_slot_state(z2->input_dev, MT_TOOL_FINGER, slot_valid)) | 
 | 			continue; | 
 | 		touchscreen_report_pos(z2->input_dev, &z2->props, | 
 | 				       le16_to_cpu(fingers[i].abs_x), | 
 | 				       le16_to_cpu(fingers[i].abs_y), | 
 | 				       true); | 
 | 		input_report_abs(z2->input_dev, ABS_MT_WIDTH_MAJOR, | 
 | 				 le16_to_cpu(fingers[i].tool_major)); | 
 | 		input_report_abs(z2->input_dev, ABS_MT_WIDTH_MINOR, | 
 | 				 le16_to_cpu(fingers[i].tool_minor)); | 
 | 		input_report_abs(z2->input_dev, ABS_MT_ORIENTATION, | 
 | 				 le16_to_cpu(fingers[i].orientation)); | 
 | 		input_report_abs(z2->input_dev, ABS_MT_TOUCH_MAJOR, | 
 | 				 le16_to_cpu(fingers[i].touch_major)); | 
 | 		input_report_abs(z2->input_dev, ABS_MT_TOUCH_MINOR, | 
 | 				 le16_to_cpu(fingers[i].touch_minor)); | 
 | 	} | 
 | 	input_mt_sync_frame(z2->input_dev); | 
 | 	input_sync(z2->input_dev); | 
 | } | 
 |  | 
 | static int apple_z2_read_packet(struct apple_z2 *z2) | 
 | { | 
 | 	struct apple_z2_read_interrupt_cmd *len_cmd = (void *)z2->tx_buf; | 
 | 	struct spi_transfer xfer; | 
 | 	int error; | 
 | 	size_t pkt_len; | 
 |  | 
 | 	memset(&xfer, 0, sizeof(xfer)); | 
 | 	len_cmd->cmd = APPLE_Z2_CMD_READ_INTERRUPT_DATA; | 
 | 	len_cmd->counter = z2->index_parity + 1; | 
 | 	len_cmd->checksum = | 
 | 		cpu_to_le16(APPLE_Z2_CMD_READ_INTERRUPT_DATA + len_cmd->counter); | 
 | 	z2->index_parity = !z2->index_parity; | 
 | 	xfer.tx_buf = z2->tx_buf; | 
 | 	xfer.rx_buf = z2->rx_buf; | 
 | 	xfer.len = sizeof(*len_cmd); | 
 |  | 
 | 	error = spi_sync_transfer(z2->spidev, &xfer, 1); | 
 | 	if (error) | 
 | 		return error; | 
 |  | 
 | 	pkt_len = (get_unaligned_le16(z2->rx_buf + 1) + 8) & 0xfffffffc; | 
 |  | 
 | 	error = spi_read(z2->spidev, z2->rx_buf, pkt_len); | 
 | 	if (error) | 
 | 		return error; | 
 |  | 
 | 	apple_z2_parse_touches(z2, z2->rx_buf + 5, pkt_len - 5); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static irqreturn_t apple_z2_irq(int irq, void *data) | 
 | { | 
 | 	struct apple_z2 *z2 = data; | 
 |  | 
 | 	if (unlikely(!z2->booted)) | 
 | 		complete(&z2->boot_irq); | 
 | 	else | 
 | 		apple_z2_read_packet(z2); | 
 |  | 
 | 	return IRQ_HANDLED; | 
 | } | 
 |  | 
 | /* Build calibration blob, caller is responsible for freeing the blob data. */ | 
 | static const u8 *apple_z2_build_cal_blob(struct apple_z2 *z2, | 
 | 					 u32 address, size_t *size) | 
 | { | 
 | 	u8 *cal_data; | 
 | 	int cal_size; | 
 | 	size_t blob_size; | 
 | 	u32 checksum; | 
 | 	u16 checksum_hdr; | 
 | 	int i; | 
 | 	struct apple_z2_hbpp_blob_hdr *hdr; | 
 | 	int error; | 
 |  | 
 | 	if (!device_property_present(&z2->spidev->dev, CAL_PROP_NAME)) | 
 | 		return NULL; | 
 |  | 
 | 	cal_size = device_property_count_u8(&z2->spidev->dev, CAL_PROP_NAME); | 
 | 	if (cal_size < 0) | 
 | 		return ERR_PTR(cal_size); | 
 |  | 
 | 	blob_size = sizeof(struct apple_z2_hbpp_blob_hdr) + cal_size + sizeof(__le32); | 
 | 	u8 *blob_data __free(kfree) = kzalloc(blob_size, GFP_KERNEL); | 
 | 	if (!blob_data) | 
 | 		return ERR_PTR(-ENOMEM); | 
 |  | 
 | 	hdr = (struct apple_z2_hbpp_blob_hdr *)blob_data; | 
 | 	hdr->cmd = cpu_to_le16(APPLE_Z2_HBPP_CMD_BLOB); | 
 | 	hdr->len = cpu_to_le16(round_up(cal_size, 4) / 4); | 
 | 	hdr->addr = cpu_to_le32(address); | 
 |  | 
 | 	checksum_hdr = 0; | 
 | 	for (i = 2; i < 8; i++) | 
 | 		checksum_hdr += blob_data[i]; | 
 | 	hdr->checksum = cpu_to_le16(checksum_hdr); | 
 |  | 
 | 	cal_data = blob_data + sizeof(struct apple_z2_hbpp_blob_hdr); | 
 | 	error = device_property_read_u8_array(&z2->spidev->dev, CAL_PROP_NAME, | 
 | 					      cal_data, cal_size); | 
 | 	if (error) | 
 | 		return ERR_PTR(error); | 
 |  | 
 | 	checksum = 0; | 
 | 	for (i = 0; i < cal_size; i++) | 
 | 		checksum += cal_data[i]; | 
 | 	put_unaligned_le32(checksum, cal_data + cal_size); | 
 |  | 
 | 	*size = blob_size; | 
 | 	return no_free_ptr(blob_data); | 
 | } | 
 |  | 
 | static int apple_z2_send_firmware_blob(struct apple_z2 *z2, const u8 *data, | 
 | 				       u32 size, bool init) | 
 | { | 
 | 	struct spi_message msg; | 
 | 	struct spi_transfer blob_xfer, ack_xfer; | 
 | 	int error; | 
 |  | 
 | 	z2->tx_buf[0] = 0x1a; | 
 | 	z2->tx_buf[1] = 0xa1; | 
 |  | 
 | 	spi_message_init(&msg); | 
 | 	memset(&blob_xfer, 0, sizeof(blob_xfer)); | 
 | 	memset(&ack_xfer, 0, sizeof(ack_xfer)); | 
 |  | 
 | 	blob_xfer.tx_buf = data; | 
 | 	blob_xfer.len = size; | 
 | 	blob_xfer.bits_per_word = init ? 8 : 16; | 
 | 	spi_message_add_tail(&blob_xfer, &msg); | 
 |  | 
 | 	ack_xfer.tx_buf = z2->tx_buf; | 
 | 	ack_xfer.len = 2; | 
 | 	spi_message_add_tail(&ack_xfer, &msg); | 
 |  | 
 | 	reinit_completion(&z2->boot_irq); | 
 | 	error = spi_sync(z2->spidev, &msg); | 
 | 	if (error) | 
 | 		return error; | 
 |  | 
 | 	/* Irq only happens sometimes, but the thing boots reliably nonetheless */ | 
 | 	wait_for_completion_timeout(&z2->boot_irq, msecs_to_jiffies(20)); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int apple_z2_upload_firmware(struct apple_z2 *z2) | 
 | { | 
 | 	const struct apple_z2_fw_hdr *fw_hdr; | 
 | 	size_t fw_idx = sizeof(struct apple_z2_fw_hdr); | 
 | 	int error; | 
 | 	u32 load_cmd; | 
 | 	u32 address; | 
 | 	bool init; | 
 | 	size_t size; | 
 |  | 
 | 	const struct firmware *fw __free(firmware) = NULL; | 
 | 	error = request_firmware(&fw, z2->fw_name, &z2->spidev->dev); | 
 | 	if (error) { | 
 | 		dev_err(&z2->spidev->dev, "unable to load firmware\n"); | 
 | 		return error; | 
 | 	} | 
 |  | 
 | 	fw_hdr = (const struct apple_z2_fw_hdr *)fw->data; | 
 | 	if (le32_to_cpu(fw_hdr->magic) != APPLE_Z2_FW_MAGIC || le32_to_cpu(fw_hdr->version) != 1) { | 
 | 		dev_err(&z2->spidev->dev, "invalid firmware header\n"); | 
 | 		return -EINVAL; | 
 | 	} | 
 |  | 
 | 	/* | 
 | 	 * This will interrupt the upload half-way if the file is malformed | 
 | 	 * As the device has no non-volatile storage to corrupt, and gets reset | 
 | 	 * on boot anyway, this is fine. | 
 | 	 */ | 
 | 	while (fw_idx < fw->size) { | 
 | 		if (fw->size - fw_idx < 8) { | 
 | 			dev_err(&z2->spidev->dev, "firmware malformed\n"); | 
 | 			return -EINVAL; | 
 | 		} | 
 |  | 
 | 		load_cmd = le32_to_cpup((__force __le32 *)(fw->data + fw_idx)); | 
 | 		fw_idx += sizeof(u32); | 
 | 		if (load_cmd == LOAD_COMMAND_INIT_PAYLOAD || load_cmd == LOAD_COMMAND_SEND_BLOB) { | 
 | 			size = le32_to_cpup((__force __le32 *)(fw->data + fw_idx)); | 
 | 			fw_idx += sizeof(u32); | 
 | 			if (fw->size - fw_idx < size) { | 
 | 				dev_err(&z2->spidev->dev, "firmware malformed\n"); | 
 | 				return -EINVAL; | 
 | 			} | 
 | 			init = load_cmd == LOAD_COMMAND_INIT_PAYLOAD; | 
 | 			error = apple_z2_send_firmware_blob(z2, fw->data + fw_idx, | 
 | 							    size, init); | 
 | 			if (error) | 
 | 				return error; | 
 | 			fw_idx += size; | 
 | 		} else if (load_cmd == LOAD_COMMAND_SEND_CALIBRATION) { | 
 | 			address = le32_to_cpup((__force __le32 *)(fw->data + fw_idx)); | 
 | 			fw_idx += sizeof(u32); | 
 |  | 
 | 			const u8 *data __free(kfree) = | 
 | 				apple_z2_build_cal_blob(z2, address, &size); | 
 | 			if (IS_ERR(data)) | 
 | 				return PTR_ERR(data); | 
 |  | 
 | 			if (data) { | 
 | 				error = apple_z2_send_firmware_blob(z2, data, size, false); | 
 | 				if (error) | 
 | 					return error; | 
 | 			} | 
 | 		} else { | 
 | 			dev_err(&z2->spidev->dev, "firmware malformed\n"); | 
 | 			return -EINVAL; | 
 | 		} | 
 | 		fw_idx = round_up(fw_idx, 4); | 
 | 	} | 
 |  | 
 |  | 
 | 	z2->booted = true; | 
 | 	apple_z2_read_packet(z2); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int apple_z2_boot(struct apple_z2 *z2) | 
 | { | 
 | 	int error; | 
 |  | 
 | 	reinit_completion(&z2->boot_irq); | 
 | 	enable_irq(z2->spidev->irq); | 
 | 	gpiod_set_value(z2->reset_gpio, 0); | 
 | 	if (!wait_for_completion_timeout(&z2->boot_irq, msecs_to_jiffies(20))) | 
 | 		return -ETIMEDOUT; | 
 |  | 
 | 	error = apple_z2_upload_firmware(z2); | 
 | 	if (error) { | 
 | 		gpiod_set_value(z2->reset_gpio, 1); | 
 | 		disable_irq(z2->spidev->irq); | 
 | 		return error; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int apple_z2_probe(struct spi_device *spi) | 
 | { | 
 | 	struct device *dev = &spi->dev; | 
 | 	struct apple_z2 *z2; | 
 | 	int error; | 
 |  | 
 | 	z2 = devm_kzalloc(dev, sizeof(*z2), GFP_KERNEL); | 
 | 	if (!z2) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	z2->tx_buf = devm_kzalloc(dev, sizeof(struct apple_z2_read_interrupt_cmd), GFP_KERNEL); | 
 | 	if (!z2->tx_buf) | 
 | 		return -ENOMEM; | 
 | 	/* 4096 will end up being rounded up to 8192 due to devres header */ | 
 | 	z2->rx_buf = devm_kzalloc(dev, 4000, GFP_KERNEL); | 
 | 	if (!z2->rx_buf) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	z2->spidev = spi; | 
 | 	init_completion(&z2->boot_irq); | 
 | 	spi_set_drvdata(spi, z2); | 
 |  | 
 | 	/* Reset the device on boot */ | 
 | 	z2->reset_gpio = devm_gpiod_get(dev, "reset", GPIOD_OUT_HIGH); | 
 | 	if (IS_ERR(z2->reset_gpio)) | 
 | 		return dev_err_probe(dev, PTR_ERR(z2->reset_gpio), "unable to get reset\n"); | 
 |  | 
 | 	error = devm_request_threaded_irq(dev, z2->spidev->irq, NULL, apple_z2_irq, | 
 | 					  IRQF_ONESHOT | IRQF_NO_AUTOEN, | 
 | 					  "apple-z2-irq", z2); | 
 | 	if (error) | 
 | 		return dev_err_probe(dev, error, "unable to request irq\n"); | 
 |  | 
 | 	error = device_property_read_string(dev, "firmware-name", &z2->fw_name); | 
 | 	if (error) | 
 | 		return dev_err_probe(dev, error, "unable to get firmware name\n"); | 
 |  | 
 | 	z2->input_dev = devm_input_allocate_device(dev); | 
 | 	if (!z2->input_dev) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	z2->input_dev->name = (char *)spi_get_device_id(spi)->driver_data; | 
 | 	z2->input_dev->phys = "apple_z2"; | 
 | 	z2->input_dev->id.bustype = BUS_SPI; | 
 |  | 
 | 	/* Allocate the axes before setting from DT */ | 
 | 	input_set_abs_params(z2->input_dev, ABS_MT_POSITION_X, 0, 0, 0, 0); | 
 | 	input_set_abs_params(z2->input_dev, ABS_MT_POSITION_Y, 0, 0, 0, 0); | 
 | 	touchscreen_parse_properties(z2->input_dev, true, &z2->props); | 
 | 	input_abs_set_res(z2->input_dev, ABS_MT_POSITION_X, 100); | 
 | 	input_abs_set_res(z2->input_dev, ABS_MT_POSITION_Y, 100); | 
 | 	input_set_abs_params(z2->input_dev, ABS_MT_WIDTH_MAJOR, 0, 65535, 0, 0); | 
 | 	input_set_abs_params(z2->input_dev, ABS_MT_WIDTH_MINOR, 0, 65535, 0, 0); | 
 | 	input_set_abs_params(z2->input_dev, ABS_MT_TOUCH_MAJOR, 0, 65535, 0, 0); | 
 | 	input_set_abs_params(z2->input_dev, ABS_MT_TOUCH_MINOR, 0, 65535, 0, 0); | 
 | 	input_set_abs_params(z2->input_dev, ABS_MT_ORIENTATION, -32768, 32767, 0, 0); | 
 |  | 
 | 	error = input_mt_init_slots(z2->input_dev, 256, INPUT_MT_DIRECT); | 
 | 	if (error) | 
 | 		return dev_err_probe(dev, error, "unable to initialize multitouch slots\n"); | 
 |  | 
 | 	error = input_register_device(z2->input_dev); | 
 | 	if (error) | 
 | 		return dev_err_probe(dev, error, "unable to register input device\n"); | 
 |  | 
 | 	/* Wait for device reset to finish */ | 
 | 	usleep_range(5000, 10000); | 
 | 	error = apple_z2_boot(z2); | 
 | 	if (error) | 
 | 		return error; | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void apple_z2_shutdown(struct spi_device *spi) | 
 | { | 
 | 	struct apple_z2 *z2 = spi_get_drvdata(spi); | 
 |  | 
 | 	disable_irq(z2->spidev->irq); | 
 | 	gpiod_direction_output(z2->reset_gpio, 1); | 
 | 	z2->booted = false; | 
 | } | 
 |  | 
 | static int apple_z2_suspend(struct device *dev) | 
 | { | 
 | 	apple_z2_shutdown(to_spi_device(dev)); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int apple_z2_resume(struct device *dev) | 
 | { | 
 | 	struct apple_z2 *z2 = spi_get_drvdata(to_spi_device(dev)); | 
 |  | 
 | 	return apple_z2_boot(z2); | 
 | } | 
 |  | 
 | static DEFINE_SIMPLE_DEV_PM_OPS(apple_z2_pm, apple_z2_suspend, apple_z2_resume); | 
 |  | 
 | static const struct of_device_id apple_z2_of_match[] = { | 
 | 	{ .compatible = "apple,j293-touchbar" }, | 
 | 	{ .compatible = "apple,j493-touchbar" }, | 
 | 	{} | 
 | }; | 
 | MODULE_DEVICE_TABLE(of, apple_z2_of_match); | 
 |  | 
 | static struct spi_device_id apple_z2_of_id[] = { | 
 | 	{ .name = "j293-touchbar", .driver_data = (kernel_ulong_t)"MacBookPro17,1 Touch Bar" }, | 
 | 	{ .name = "j493-touchbar", .driver_data = (kernel_ulong_t)"Mac14,7 Touch Bar" }, | 
 | 	{} | 
 | }; | 
 | MODULE_DEVICE_TABLE(spi, apple_z2_of_id); | 
 |  | 
 | static struct spi_driver apple_z2_driver = { | 
 | 	.driver = { | 
 | 		.name	= "apple-z2", | 
 | 		.pm	= pm_sleep_ptr(&apple_z2_pm), | 
 | 		.of_match_table = apple_z2_of_match, | 
 | 		.probe_type = PROBE_PREFER_ASYNCHRONOUS, | 
 | 	}, | 
 | 	.id_table = apple_z2_of_id, | 
 | 	.probe    = apple_z2_probe, | 
 | 	.remove   = apple_z2_shutdown, | 
 | }; | 
 |  | 
 | module_spi_driver(apple_z2_driver); | 
 |  | 
 | MODULE_LICENSE("GPL"); | 
 | MODULE_FIRMWARE("apple/dfrmtfw-*.bin"); | 
 | MODULE_DESCRIPTION("Apple Z2 touchscreens driver"); |