10670c2f5SKerem Karabay // SPDX-License-Identifier: GPL-2.0 20670c2f5SKerem Karabay /* 30670c2f5SKerem Karabay * Apple Touch Bar DRM Driver 40670c2f5SKerem Karabay * 50670c2f5SKerem Karabay * Copyright (c) 2023 Kerem Karabay <kekrby@gmail.com> 60670c2f5SKerem Karabay */ 70670c2f5SKerem Karabay 80670c2f5SKerem Karabay #include <linux/align.h> 90670c2f5SKerem Karabay #include <linux/array_size.h> 100670c2f5SKerem Karabay #include <linux/bitops.h> 110670c2f5SKerem Karabay #include <linux/bug.h> 120670c2f5SKerem Karabay #include <linux/container_of.h> 130670c2f5SKerem Karabay #include <linux/err.h> 140670c2f5SKerem Karabay #include <linux/module.h> 150670c2f5SKerem Karabay #include <linux/overflow.h> 160670c2f5SKerem Karabay #include <linux/slab.h> 170670c2f5SKerem Karabay #include <linux/types.h> 180670c2f5SKerem Karabay #include <linux/unaligned.h> 190670c2f5SKerem Karabay #include <linux/usb.h> 200670c2f5SKerem Karabay 210670c2f5SKerem Karabay #include <drm/drm_atomic.h> 220670c2f5SKerem Karabay #include <drm/drm_atomic_helper.h> 230670c2f5SKerem Karabay #include <drm/drm_crtc.h> 240670c2f5SKerem Karabay #include <drm/drm_damage_helper.h> 250670c2f5SKerem Karabay #include <drm/drm_drv.h> 260670c2f5SKerem Karabay #include <drm/drm_encoder.h> 270670c2f5SKerem Karabay #include <drm/drm_format_helper.h> 280670c2f5SKerem Karabay #include <drm/drm_fourcc.h> 290670c2f5SKerem Karabay #include <drm/drm_framebuffer.h> 300670c2f5SKerem Karabay #include <drm/drm_gem_atomic_helper.h> 310670c2f5SKerem Karabay #include <drm/drm_gem_framebuffer_helper.h> 320670c2f5SKerem Karabay #include <drm/drm_gem_shmem_helper.h> 330670c2f5SKerem Karabay #include <drm/drm_plane.h> 340670c2f5SKerem Karabay #include <drm/drm_print.h> 350670c2f5SKerem Karabay #include <drm/drm_probe_helper.h> 360670c2f5SKerem Karabay 370670c2f5SKerem Karabay #define APPLETBDRM_PIXEL_FORMAT cpu_to_le32(0x52474241) /* RGBA, the actual format is BGR888 */ 380670c2f5SKerem Karabay #define APPLETBDRM_BITS_PER_PIXEL 24 390670c2f5SKerem Karabay 400670c2f5SKerem Karabay #define APPLETBDRM_MSG_CLEAR_DISPLAY cpu_to_le32(0x434c5244) /* CLRD */ 410670c2f5SKerem Karabay #define APPLETBDRM_MSG_GET_INFORMATION cpu_to_le32(0x47494e46) /* GINF */ 420670c2f5SKerem Karabay #define APPLETBDRM_MSG_UPDATE_COMPLETE cpu_to_le32(0x5544434c) /* UDCL */ 430670c2f5SKerem Karabay #define APPLETBDRM_MSG_SIGNAL_READINESS cpu_to_le32(0x52454459) /* REDY */ 440670c2f5SKerem Karabay 450670c2f5SKerem Karabay #define APPLETBDRM_BULK_MSG_TIMEOUT 1000 460670c2f5SKerem Karabay 470670c2f5SKerem Karabay #define drm_to_adev(_drm) container_of(_drm, struct appletbdrm_device, drm) 480670c2f5SKerem Karabay #define adev_to_udev(adev) interface_to_usbdev(to_usb_interface(adev->dmadev)) 490670c2f5SKerem Karabay 500670c2f5SKerem Karabay struct appletbdrm_msg_request_header { 510670c2f5SKerem Karabay __le16 unk_00; 520670c2f5SKerem Karabay __le16 unk_02; 530670c2f5SKerem Karabay __le32 unk_04; 540670c2f5SKerem Karabay __le32 unk_08; 550670c2f5SKerem Karabay __le32 size; 560670c2f5SKerem Karabay } __packed; 570670c2f5SKerem Karabay 580670c2f5SKerem Karabay struct appletbdrm_msg_response_header { 590670c2f5SKerem Karabay u8 unk_00[16]; 600670c2f5SKerem Karabay __le32 msg; 610670c2f5SKerem Karabay } __packed; 620670c2f5SKerem Karabay 630670c2f5SKerem Karabay struct appletbdrm_msg_simple_request { 640670c2f5SKerem Karabay struct appletbdrm_msg_request_header header; 650670c2f5SKerem Karabay __le32 msg; 660670c2f5SKerem Karabay u8 unk_14[8]; 670670c2f5SKerem Karabay __le32 size; 680670c2f5SKerem Karabay } __packed; 690670c2f5SKerem Karabay 700670c2f5SKerem Karabay struct appletbdrm_msg_information { 710670c2f5SKerem Karabay struct appletbdrm_msg_response_header header; 720670c2f5SKerem Karabay u8 unk_14[12]; 730670c2f5SKerem Karabay __le32 width; 740670c2f5SKerem Karabay __le32 height; 750670c2f5SKerem Karabay u8 bits_per_pixel; 760670c2f5SKerem Karabay __le32 bytes_per_row; 770670c2f5SKerem Karabay __le32 orientation; 780670c2f5SKerem Karabay __le32 bitmap_info; 790670c2f5SKerem Karabay __le32 pixel_format; 800670c2f5SKerem Karabay __le32 width_inches; /* floating point */ 810670c2f5SKerem Karabay __le32 height_inches; /* floating point */ 820670c2f5SKerem Karabay } __packed; 830670c2f5SKerem Karabay 840670c2f5SKerem Karabay struct appletbdrm_frame { 850670c2f5SKerem Karabay __le16 begin_x; 860670c2f5SKerem Karabay __le16 begin_y; 870670c2f5SKerem Karabay __le16 width; 880670c2f5SKerem Karabay __le16 height; 890670c2f5SKerem Karabay __le32 buf_size; 900670c2f5SKerem Karabay u8 buf[]; 910670c2f5SKerem Karabay } __packed; 920670c2f5SKerem Karabay 930670c2f5SKerem Karabay struct appletbdrm_fb_request_footer { 940670c2f5SKerem Karabay u8 unk_00[12]; 950670c2f5SKerem Karabay __le32 unk_0c; 960670c2f5SKerem Karabay u8 unk_10[12]; 970670c2f5SKerem Karabay __le32 unk_1c; 980670c2f5SKerem Karabay __le64 timestamp; 990670c2f5SKerem Karabay u8 unk_28[12]; 1000670c2f5SKerem Karabay __le32 unk_34; 1010670c2f5SKerem Karabay u8 unk_38[20]; 1020670c2f5SKerem Karabay __le32 unk_4c; 1030670c2f5SKerem Karabay } __packed; 1040670c2f5SKerem Karabay 1050670c2f5SKerem Karabay struct appletbdrm_fb_request { 1060670c2f5SKerem Karabay struct appletbdrm_msg_request_header header; 1070670c2f5SKerem Karabay __le16 unk_10; 1080670c2f5SKerem Karabay u8 msg_id; 1090670c2f5SKerem Karabay u8 unk_13[29]; 1100670c2f5SKerem Karabay /* 1110670c2f5SKerem Karabay * Contents of `data`: 1120670c2f5SKerem Karabay * - struct appletbdrm_frame frames[]; 1130670c2f5SKerem Karabay * - struct appletbdrm_fb_request_footer footer; 1140670c2f5SKerem Karabay * - padding to make the total size a multiple of 16 1150670c2f5SKerem Karabay */ 1160670c2f5SKerem Karabay u8 data[]; 1170670c2f5SKerem Karabay } __packed; 1180670c2f5SKerem Karabay 1190670c2f5SKerem Karabay struct appletbdrm_fb_request_response { 1200670c2f5SKerem Karabay struct appletbdrm_msg_response_header header; 1210670c2f5SKerem Karabay u8 unk_14[12]; 1220670c2f5SKerem Karabay __le64 timestamp; 1230670c2f5SKerem Karabay } __packed; 1240670c2f5SKerem Karabay 1250670c2f5SKerem Karabay struct appletbdrm_device { 1260670c2f5SKerem Karabay struct device *dmadev; 1270670c2f5SKerem Karabay 1280670c2f5SKerem Karabay unsigned int in_ep; 1290670c2f5SKerem Karabay unsigned int out_ep; 1300670c2f5SKerem Karabay 1310670c2f5SKerem Karabay unsigned int width; 1320670c2f5SKerem Karabay unsigned int height; 1330670c2f5SKerem Karabay 1340670c2f5SKerem Karabay struct drm_device drm; 1350670c2f5SKerem Karabay struct drm_display_mode mode; 1360670c2f5SKerem Karabay struct drm_connector connector; 1370670c2f5SKerem Karabay struct drm_plane primary_plane; 1380670c2f5SKerem Karabay struct drm_crtc crtc; 1390670c2f5SKerem Karabay struct drm_encoder encoder; 1400670c2f5SKerem Karabay }; 1410670c2f5SKerem Karabay 1420670c2f5SKerem Karabay struct appletbdrm_plane_state { 1430670c2f5SKerem Karabay struct drm_shadow_plane_state base; 1440670c2f5SKerem Karabay struct appletbdrm_fb_request *request; 1450670c2f5SKerem Karabay struct appletbdrm_fb_request_response *response; 1460670c2f5SKerem Karabay size_t request_size; 1470670c2f5SKerem Karabay size_t frames_size; 1480670c2f5SKerem Karabay }; 1490670c2f5SKerem Karabay 1500670c2f5SKerem Karabay static inline struct appletbdrm_plane_state *to_appletbdrm_plane_state(struct drm_plane_state *state) 1510670c2f5SKerem Karabay { 1520670c2f5SKerem Karabay return container_of(state, struct appletbdrm_plane_state, base.base); 1530670c2f5SKerem Karabay } 1540670c2f5SKerem Karabay 1550670c2f5SKerem Karabay static int appletbdrm_send_request(struct appletbdrm_device *adev, 1560670c2f5SKerem Karabay struct appletbdrm_msg_request_header *request, size_t size) 1570670c2f5SKerem Karabay { 1580670c2f5SKerem Karabay struct usb_device *udev = adev_to_udev(adev); 1590670c2f5SKerem Karabay struct drm_device *drm = &adev->drm; 1600670c2f5SKerem Karabay int ret, actual_size; 1610670c2f5SKerem Karabay 1620670c2f5SKerem Karabay ret = usb_bulk_msg(udev, usb_sndbulkpipe(udev, adev->out_ep), 1630670c2f5SKerem Karabay request, size, &actual_size, APPLETBDRM_BULK_MSG_TIMEOUT); 1640670c2f5SKerem Karabay if (ret) { 1650670c2f5SKerem Karabay drm_err(drm, "Failed to send message (%d)\n", ret); 1660670c2f5SKerem Karabay return ret; 1670670c2f5SKerem Karabay } 1680670c2f5SKerem Karabay 1690670c2f5SKerem Karabay if (actual_size != size) { 170*c449f506SNathan Chancellor drm_err(drm, "Actual size (%d) doesn't match expected size (%zu)\n", 1710670c2f5SKerem Karabay actual_size, size); 1720670c2f5SKerem Karabay return -EIO; 1730670c2f5SKerem Karabay } 1740670c2f5SKerem Karabay 1750670c2f5SKerem Karabay return 0; 1760670c2f5SKerem Karabay } 1770670c2f5SKerem Karabay 1780670c2f5SKerem Karabay static int appletbdrm_read_response(struct appletbdrm_device *adev, 1790670c2f5SKerem Karabay struct appletbdrm_msg_response_header *response, 1800670c2f5SKerem Karabay size_t size, __le32 expected_response) 1810670c2f5SKerem Karabay { 1820670c2f5SKerem Karabay struct usb_device *udev = adev_to_udev(adev); 1830670c2f5SKerem Karabay struct drm_device *drm = &adev->drm; 1840670c2f5SKerem Karabay int ret, actual_size; 1850670c2f5SKerem Karabay bool readiness_signal_received = false; 1860670c2f5SKerem Karabay 1870670c2f5SKerem Karabay retry: 1880670c2f5SKerem Karabay ret = usb_bulk_msg(udev, usb_rcvbulkpipe(udev, adev->in_ep), 1890670c2f5SKerem Karabay response, size, &actual_size, APPLETBDRM_BULK_MSG_TIMEOUT); 1900670c2f5SKerem Karabay if (ret) { 1910670c2f5SKerem Karabay drm_err(drm, "Failed to read response (%d)\n", ret); 1920670c2f5SKerem Karabay return ret; 1930670c2f5SKerem Karabay } 1940670c2f5SKerem Karabay 1950670c2f5SKerem Karabay /* 1960670c2f5SKerem Karabay * The device responds to the first request sent in a particular 1970670c2f5SKerem Karabay * timeframe after the USB device configuration is set with a readiness 1980670c2f5SKerem Karabay * signal, in which case the response should be read again 1990670c2f5SKerem Karabay */ 2000670c2f5SKerem Karabay if (response->msg == APPLETBDRM_MSG_SIGNAL_READINESS) { 2010670c2f5SKerem Karabay if (!readiness_signal_received) { 2020670c2f5SKerem Karabay readiness_signal_received = true; 2030670c2f5SKerem Karabay goto retry; 2040670c2f5SKerem Karabay } 2050670c2f5SKerem Karabay 2060670c2f5SKerem Karabay drm_err(drm, "Encountered unexpected readiness signal\n"); 2070670c2f5SKerem Karabay return -EINTR; 2080670c2f5SKerem Karabay } 2090670c2f5SKerem Karabay 2100670c2f5SKerem Karabay if (actual_size != size) { 211*c449f506SNathan Chancellor drm_err(drm, "Actual size (%d) doesn't match expected size (%zu)\n", 2120670c2f5SKerem Karabay actual_size, size); 2130670c2f5SKerem Karabay return -EBADMSG; 2140670c2f5SKerem Karabay } 2150670c2f5SKerem Karabay 2160670c2f5SKerem Karabay if (response->msg != expected_response) { 2170670c2f5SKerem Karabay drm_err(drm, "Unexpected response from device (expected %p4cc found %p4cc)\n", 2180670c2f5SKerem Karabay &expected_response, &response->msg); 2190670c2f5SKerem Karabay return -EIO; 2200670c2f5SKerem Karabay } 2210670c2f5SKerem Karabay 2220670c2f5SKerem Karabay return 0; 2230670c2f5SKerem Karabay } 2240670c2f5SKerem Karabay 2250670c2f5SKerem Karabay static int appletbdrm_send_msg(struct appletbdrm_device *adev, __le32 msg) 2260670c2f5SKerem Karabay { 2270670c2f5SKerem Karabay struct appletbdrm_msg_simple_request *request; 2280670c2f5SKerem Karabay int ret; 2290670c2f5SKerem Karabay 2300670c2f5SKerem Karabay request = kzalloc(sizeof(*request), GFP_KERNEL); 2310670c2f5SKerem Karabay if (!request) 2320670c2f5SKerem Karabay return -ENOMEM; 2330670c2f5SKerem Karabay 2340670c2f5SKerem Karabay request->header.unk_00 = cpu_to_le16(2); 2350670c2f5SKerem Karabay request->header.unk_02 = cpu_to_le16(0x1512); 2360670c2f5SKerem Karabay request->header.size = cpu_to_le32(sizeof(*request) - sizeof(request->header)); 2370670c2f5SKerem Karabay request->msg = msg; 2380670c2f5SKerem Karabay request->size = request->header.size; 2390670c2f5SKerem Karabay 2400670c2f5SKerem Karabay ret = appletbdrm_send_request(adev, &request->header, sizeof(*request)); 2410670c2f5SKerem Karabay 2420670c2f5SKerem Karabay kfree(request); 2430670c2f5SKerem Karabay 2440670c2f5SKerem Karabay return ret; 2450670c2f5SKerem Karabay } 2460670c2f5SKerem Karabay 2470670c2f5SKerem Karabay static int appletbdrm_clear_display(struct appletbdrm_device *adev) 2480670c2f5SKerem Karabay { 2490670c2f5SKerem Karabay return appletbdrm_send_msg(adev, APPLETBDRM_MSG_CLEAR_DISPLAY); 2500670c2f5SKerem Karabay } 2510670c2f5SKerem Karabay 2520670c2f5SKerem Karabay static int appletbdrm_signal_readiness(struct appletbdrm_device *adev) 2530670c2f5SKerem Karabay { 2540670c2f5SKerem Karabay return appletbdrm_send_msg(adev, APPLETBDRM_MSG_SIGNAL_READINESS); 2550670c2f5SKerem Karabay } 2560670c2f5SKerem Karabay 2570670c2f5SKerem Karabay static int appletbdrm_get_information(struct appletbdrm_device *adev) 2580670c2f5SKerem Karabay { 2590670c2f5SKerem Karabay struct appletbdrm_msg_information *info; 2600670c2f5SKerem Karabay struct drm_device *drm = &adev->drm; 2610670c2f5SKerem Karabay u8 bits_per_pixel; 2620670c2f5SKerem Karabay __le32 pixel_format; 2630670c2f5SKerem Karabay int ret; 2640670c2f5SKerem Karabay 2650670c2f5SKerem Karabay info = kzalloc(sizeof(*info), GFP_KERNEL); 2660670c2f5SKerem Karabay if (!info) 2670670c2f5SKerem Karabay return -ENOMEM; 2680670c2f5SKerem Karabay 2690670c2f5SKerem Karabay ret = appletbdrm_send_msg(adev, APPLETBDRM_MSG_GET_INFORMATION); 2700670c2f5SKerem Karabay if (ret) 2710670c2f5SKerem Karabay return ret; 2720670c2f5SKerem Karabay 2730670c2f5SKerem Karabay ret = appletbdrm_read_response(adev, &info->header, sizeof(*info), 2740670c2f5SKerem Karabay APPLETBDRM_MSG_GET_INFORMATION); 2750670c2f5SKerem Karabay if (ret) 2760670c2f5SKerem Karabay goto free_info; 2770670c2f5SKerem Karabay 2780670c2f5SKerem Karabay bits_per_pixel = info->bits_per_pixel; 2790670c2f5SKerem Karabay pixel_format = get_unaligned(&info->pixel_format); 2800670c2f5SKerem Karabay 2810670c2f5SKerem Karabay adev->width = get_unaligned_le32(&info->width); 2820670c2f5SKerem Karabay adev->height = get_unaligned_le32(&info->height); 2830670c2f5SKerem Karabay 2840670c2f5SKerem Karabay if (bits_per_pixel != APPLETBDRM_BITS_PER_PIXEL) { 2850670c2f5SKerem Karabay drm_err(drm, "Encountered unexpected bits per pixel value (%d)\n", bits_per_pixel); 2860670c2f5SKerem Karabay ret = -EINVAL; 2870670c2f5SKerem Karabay goto free_info; 2880670c2f5SKerem Karabay } 2890670c2f5SKerem Karabay 2900670c2f5SKerem Karabay if (pixel_format != APPLETBDRM_PIXEL_FORMAT) { 2910670c2f5SKerem Karabay drm_err(drm, "Encountered unknown pixel format (%p4cc)\n", &pixel_format); 2920670c2f5SKerem Karabay ret = -EINVAL; 2930670c2f5SKerem Karabay goto free_info; 2940670c2f5SKerem Karabay } 2950670c2f5SKerem Karabay 2960670c2f5SKerem Karabay free_info: 2970670c2f5SKerem Karabay kfree(info); 2980670c2f5SKerem Karabay 2990670c2f5SKerem Karabay return ret; 3000670c2f5SKerem Karabay } 3010670c2f5SKerem Karabay 3020670c2f5SKerem Karabay static u32 rect_size(struct drm_rect *rect) 3030670c2f5SKerem Karabay { 3040670c2f5SKerem Karabay return drm_rect_width(rect) * drm_rect_height(rect) * 3050670c2f5SKerem Karabay (BITS_TO_BYTES(APPLETBDRM_BITS_PER_PIXEL)); 3060670c2f5SKerem Karabay } 3070670c2f5SKerem Karabay 3080670c2f5SKerem Karabay static int appletbdrm_connector_helper_get_modes(struct drm_connector *connector) 3090670c2f5SKerem Karabay { 3100670c2f5SKerem Karabay struct appletbdrm_device *adev = drm_to_adev(connector->dev); 3110670c2f5SKerem Karabay 3120670c2f5SKerem Karabay return drm_connector_helper_get_modes_fixed(connector, &adev->mode); 3130670c2f5SKerem Karabay } 3140670c2f5SKerem Karabay 3150670c2f5SKerem Karabay static const u32 appletbdrm_primary_plane_formats[] = { 3160670c2f5SKerem Karabay DRM_FORMAT_BGR888, 3170670c2f5SKerem Karabay DRM_FORMAT_XRGB8888, /* emulated */ 3180670c2f5SKerem Karabay }; 3190670c2f5SKerem Karabay 3200670c2f5SKerem Karabay static int appletbdrm_primary_plane_helper_atomic_check(struct drm_plane *plane, 3210670c2f5SKerem Karabay struct drm_atomic_state *state) 3220670c2f5SKerem Karabay { 3230670c2f5SKerem Karabay struct drm_plane_state *new_plane_state = drm_atomic_get_new_plane_state(state, plane); 3240670c2f5SKerem Karabay struct drm_plane_state *old_plane_state = drm_atomic_get_old_plane_state(state, plane); 3250670c2f5SKerem Karabay struct drm_crtc *new_crtc = new_plane_state->crtc; 3260670c2f5SKerem Karabay struct drm_crtc_state *new_crtc_state = NULL; 3270670c2f5SKerem Karabay struct appletbdrm_plane_state *appletbdrm_state = to_appletbdrm_plane_state(new_plane_state); 3280670c2f5SKerem Karabay struct drm_atomic_helper_damage_iter iter; 3290670c2f5SKerem Karabay struct drm_rect damage; 3300670c2f5SKerem Karabay size_t frames_size = 0; 3310670c2f5SKerem Karabay size_t request_size; 3320670c2f5SKerem Karabay int ret; 3330670c2f5SKerem Karabay 3340670c2f5SKerem Karabay if (new_crtc) 3350670c2f5SKerem Karabay new_crtc_state = drm_atomic_get_new_crtc_state(state, new_crtc); 3360670c2f5SKerem Karabay 3370670c2f5SKerem Karabay ret = drm_atomic_helper_check_plane_state(new_plane_state, new_crtc_state, 3380670c2f5SKerem Karabay DRM_PLANE_NO_SCALING, 3390670c2f5SKerem Karabay DRM_PLANE_NO_SCALING, 3400670c2f5SKerem Karabay false, false); 3410670c2f5SKerem Karabay if (ret) 3420670c2f5SKerem Karabay return ret; 3430670c2f5SKerem Karabay else if (!new_plane_state->visible) 3440670c2f5SKerem Karabay return 0; 3450670c2f5SKerem Karabay 3460670c2f5SKerem Karabay drm_atomic_helper_damage_iter_init(&iter, old_plane_state, new_plane_state); 3470670c2f5SKerem Karabay drm_atomic_for_each_plane_damage(&iter, &damage) { 3480670c2f5SKerem Karabay frames_size += struct_size((struct appletbdrm_frame *)0, buf, rect_size(&damage)); 3490670c2f5SKerem Karabay } 3500670c2f5SKerem Karabay 3510670c2f5SKerem Karabay if (!frames_size) 3520670c2f5SKerem Karabay return 0; 3530670c2f5SKerem Karabay 3540670c2f5SKerem Karabay request_size = ALIGN(sizeof(struct appletbdrm_fb_request) + 3550670c2f5SKerem Karabay frames_size + 3560670c2f5SKerem Karabay sizeof(struct appletbdrm_fb_request_footer), 16); 3570670c2f5SKerem Karabay 3580670c2f5SKerem Karabay appletbdrm_state->request = kzalloc(request_size, GFP_KERNEL); 3590670c2f5SKerem Karabay 3600670c2f5SKerem Karabay if (!appletbdrm_state->request) 3610670c2f5SKerem Karabay return -ENOMEM; 3620670c2f5SKerem Karabay 3630670c2f5SKerem Karabay appletbdrm_state->response = kzalloc(sizeof(*appletbdrm_state->response), GFP_KERNEL); 3640670c2f5SKerem Karabay 3650670c2f5SKerem Karabay if (!appletbdrm_state->response) 3660670c2f5SKerem Karabay return -ENOMEM; 3670670c2f5SKerem Karabay 3680670c2f5SKerem Karabay appletbdrm_state->request_size = request_size; 3690670c2f5SKerem Karabay appletbdrm_state->frames_size = frames_size; 3700670c2f5SKerem Karabay 3710670c2f5SKerem Karabay return 0; 3720670c2f5SKerem Karabay } 3730670c2f5SKerem Karabay 3740670c2f5SKerem Karabay static int appletbdrm_flush_damage(struct appletbdrm_device *adev, 3750670c2f5SKerem Karabay struct drm_plane_state *old_state, 3760670c2f5SKerem Karabay struct drm_plane_state *state) 3770670c2f5SKerem Karabay { 3780670c2f5SKerem Karabay struct appletbdrm_plane_state *appletbdrm_state = to_appletbdrm_plane_state(state); 3790670c2f5SKerem Karabay struct drm_shadow_plane_state *shadow_plane_state = to_drm_shadow_plane_state(state); 3800670c2f5SKerem Karabay struct appletbdrm_fb_request_response *response = appletbdrm_state->response; 3810670c2f5SKerem Karabay struct appletbdrm_fb_request_footer *footer; 3820670c2f5SKerem Karabay struct drm_atomic_helper_damage_iter iter; 3830670c2f5SKerem Karabay struct drm_framebuffer *fb = state->fb; 3840670c2f5SKerem Karabay struct appletbdrm_fb_request *request = appletbdrm_state->request; 3850670c2f5SKerem Karabay struct drm_device *drm = &adev->drm; 3860670c2f5SKerem Karabay struct appletbdrm_frame *frame; 3870670c2f5SKerem Karabay u64 timestamp = ktime_get_ns(); 3880670c2f5SKerem Karabay struct drm_rect damage; 3890670c2f5SKerem Karabay size_t frames_size = appletbdrm_state->frames_size; 3900670c2f5SKerem Karabay size_t request_size = appletbdrm_state->request_size; 3910670c2f5SKerem Karabay int ret; 3920670c2f5SKerem Karabay 3930670c2f5SKerem Karabay if (!frames_size) 3940670c2f5SKerem Karabay return 0; 3950670c2f5SKerem Karabay 3960670c2f5SKerem Karabay ret = drm_gem_fb_begin_cpu_access(fb, DMA_FROM_DEVICE); 3970670c2f5SKerem Karabay if (ret) { 3980670c2f5SKerem Karabay drm_err(drm, "Failed to start CPU framebuffer access (%d)\n", ret); 3990670c2f5SKerem Karabay goto end_fb_cpu_access; 4000670c2f5SKerem Karabay } 4010670c2f5SKerem Karabay 4020670c2f5SKerem Karabay request->header.unk_00 = cpu_to_le16(2); 4030670c2f5SKerem Karabay request->header.unk_02 = cpu_to_le16(0x12); 4040670c2f5SKerem Karabay request->header.unk_04 = cpu_to_le32(9); 4050670c2f5SKerem Karabay request->header.size = cpu_to_le32(request_size - sizeof(request->header)); 4060670c2f5SKerem Karabay request->unk_10 = cpu_to_le16(1); 4070670c2f5SKerem Karabay request->msg_id = timestamp; 4080670c2f5SKerem Karabay 4090670c2f5SKerem Karabay frame = (struct appletbdrm_frame *)request->data; 4100670c2f5SKerem Karabay 4110670c2f5SKerem Karabay drm_atomic_helper_damage_iter_init(&iter, old_state, state); 4120670c2f5SKerem Karabay drm_atomic_for_each_plane_damage(&iter, &damage) { 4130670c2f5SKerem Karabay struct drm_rect dst_clip = state->dst; 4140670c2f5SKerem Karabay struct iosys_map dst = IOSYS_MAP_INIT_VADDR(frame->buf); 4150670c2f5SKerem Karabay u32 buf_size = rect_size(&damage); 4160670c2f5SKerem Karabay 4170670c2f5SKerem Karabay if (!drm_rect_intersect(&dst_clip, &damage)) 4180670c2f5SKerem Karabay continue; 4190670c2f5SKerem Karabay 4200670c2f5SKerem Karabay /* 4210670c2f5SKerem Karabay * The coordinates need to be translated to the coordinate 4220670c2f5SKerem Karabay * system the device expects, see the comment in 4230670c2f5SKerem Karabay * appletbdrm_setup_mode_config 4240670c2f5SKerem Karabay */ 4250670c2f5SKerem Karabay frame->begin_x = cpu_to_le16(damage.y1); 4260670c2f5SKerem Karabay frame->begin_y = cpu_to_le16(adev->height - damage.x2); 4270670c2f5SKerem Karabay frame->width = cpu_to_le16(drm_rect_height(&damage)); 4280670c2f5SKerem Karabay frame->height = cpu_to_le16(drm_rect_width(&damage)); 4290670c2f5SKerem Karabay frame->buf_size = cpu_to_le32(buf_size); 4300670c2f5SKerem Karabay 4310670c2f5SKerem Karabay switch (fb->format->format) { 4320670c2f5SKerem Karabay case DRM_FORMAT_XRGB8888: 4330670c2f5SKerem Karabay drm_fb_xrgb8888_to_bgr888(&dst, NULL, &shadow_plane_state->data[0], fb, &damage, &shadow_plane_state->fmtcnv_state); 4340670c2f5SKerem Karabay break; 4350670c2f5SKerem Karabay default: 4360670c2f5SKerem Karabay drm_fb_memcpy(&dst, NULL, &shadow_plane_state->data[0], fb, &damage); 4370670c2f5SKerem Karabay break; 4380670c2f5SKerem Karabay } 4390670c2f5SKerem Karabay 4400670c2f5SKerem Karabay frame = (void *)frame + struct_size(frame, buf, buf_size); 4410670c2f5SKerem Karabay } 4420670c2f5SKerem Karabay 4430670c2f5SKerem Karabay footer = (struct appletbdrm_fb_request_footer *)&request->data[frames_size]; 4440670c2f5SKerem Karabay 4450670c2f5SKerem Karabay footer->unk_0c = cpu_to_le32(0xfffe); 4460670c2f5SKerem Karabay footer->unk_1c = cpu_to_le32(0x80001); 4470670c2f5SKerem Karabay footer->unk_34 = cpu_to_le32(0x80002); 4480670c2f5SKerem Karabay footer->unk_4c = cpu_to_le32(0xffff); 4490670c2f5SKerem Karabay footer->timestamp = cpu_to_le64(timestamp); 4500670c2f5SKerem Karabay 4510670c2f5SKerem Karabay ret = appletbdrm_send_request(adev, &request->header, request_size); 4520670c2f5SKerem Karabay if (ret) 4530670c2f5SKerem Karabay goto end_fb_cpu_access; 4540670c2f5SKerem Karabay 4550670c2f5SKerem Karabay ret = appletbdrm_read_response(adev, &response->header, sizeof(*response), 4560670c2f5SKerem Karabay APPLETBDRM_MSG_UPDATE_COMPLETE); 4570670c2f5SKerem Karabay if (ret) 4580670c2f5SKerem Karabay goto end_fb_cpu_access; 4590670c2f5SKerem Karabay 4600670c2f5SKerem Karabay if (response->timestamp != footer->timestamp) { 4610670c2f5SKerem Karabay drm_err(drm, "Response timestamp (%llu) doesn't match request timestamp (%llu)\n", 4620670c2f5SKerem Karabay le64_to_cpu(response->timestamp), timestamp); 4630670c2f5SKerem Karabay goto end_fb_cpu_access; 4640670c2f5SKerem Karabay } 4650670c2f5SKerem Karabay 4660670c2f5SKerem Karabay end_fb_cpu_access: 4670670c2f5SKerem Karabay drm_gem_fb_end_cpu_access(fb, DMA_FROM_DEVICE); 4680670c2f5SKerem Karabay 4690670c2f5SKerem Karabay return ret; 4700670c2f5SKerem Karabay } 4710670c2f5SKerem Karabay 4720670c2f5SKerem Karabay static void appletbdrm_primary_plane_helper_atomic_update(struct drm_plane *plane, 4730670c2f5SKerem Karabay struct drm_atomic_state *old_state) 4740670c2f5SKerem Karabay { 4750670c2f5SKerem Karabay struct appletbdrm_device *adev = drm_to_adev(plane->dev); 4760670c2f5SKerem Karabay struct drm_device *drm = plane->dev; 4770670c2f5SKerem Karabay struct drm_plane_state *plane_state = plane->state; 4780670c2f5SKerem Karabay struct drm_plane_state *old_plane_state = drm_atomic_get_old_plane_state(old_state, plane); 4790670c2f5SKerem Karabay int idx; 4800670c2f5SKerem Karabay 4810670c2f5SKerem Karabay if (!drm_dev_enter(drm, &idx)) 4820670c2f5SKerem Karabay return; 4830670c2f5SKerem Karabay 4840670c2f5SKerem Karabay appletbdrm_flush_damage(adev, old_plane_state, plane_state); 4850670c2f5SKerem Karabay 4860670c2f5SKerem Karabay drm_dev_exit(idx); 4870670c2f5SKerem Karabay } 4880670c2f5SKerem Karabay 4890670c2f5SKerem Karabay static void appletbdrm_primary_plane_helper_atomic_disable(struct drm_plane *plane, 4900670c2f5SKerem Karabay struct drm_atomic_state *state) 4910670c2f5SKerem Karabay { 4920670c2f5SKerem Karabay struct drm_device *dev = plane->dev; 4930670c2f5SKerem Karabay struct appletbdrm_device *adev = drm_to_adev(dev); 4940670c2f5SKerem Karabay int idx; 4950670c2f5SKerem Karabay 4960670c2f5SKerem Karabay if (!drm_dev_enter(dev, &idx)) 4970670c2f5SKerem Karabay return; 4980670c2f5SKerem Karabay 4990670c2f5SKerem Karabay appletbdrm_clear_display(adev); 5000670c2f5SKerem Karabay 5010670c2f5SKerem Karabay drm_dev_exit(idx); 5020670c2f5SKerem Karabay } 5030670c2f5SKerem Karabay 5040670c2f5SKerem Karabay static void appletbdrm_primary_plane_reset(struct drm_plane *plane) 5050670c2f5SKerem Karabay { 5060670c2f5SKerem Karabay struct appletbdrm_plane_state *appletbdrm_state; 5070670c2f5SKerem Karabay 5080670c2f5SKerem Karabay WARN_ON(plane->state); 5090670c2f5SKerem Karabay 5100670c2f5SKerem Karabay appletbdrm_state = kzalloc(sizeof(*appletbdrm_state), GFP_KERNEL); 5110670c2f5SKerem Karabay if (!appletbdrm_state) 5120670c2f5SKerem Karabay return; 5130670c2f5SKerem Karabay 5140670c2f5SKerem Karabay __drm_gem_reset_shadow_plane(plane, &appletbdrm_state->base); 5150670c2f5SKerem Karabay } 5160670c2f5SKerem Karabay 5170670c2f5SKerem Karabay static struct drm_plane_state *appletbdrm_primary_plane_duplicate_state(struct drm_plane *plane) 5180670c2f5SKerem Karabay { 5190670c2f5SKerem Karabay struct drm_shadow_plane_state *new_shadow_plane_state; 5200670c2f5SKerem Karabay struct appletbdrm_plane_state *appletbdrm_state; 5210670c2f5SKerem Karabay 5220670c2f5SKerem Karabay if (WARN_ON(!plane->state)) 5230670c2f5SKerem Karabay return NULL; 5240670c2f5SKerem Karabay 5250670c2f5SKerem Karabay appletbdrm_state = kzalloc(sizeof(*appletbdrm_state), GFP_KERNEL); 5260670c2f5SKerem Karabay if (!appletbdrm_state) 5270670c2f5SKerem Karabay return NULL; 5280670c2f5SKerem Karabay 5290670c2f5SKerem Karabay /* Request and response are not duplicated and are allocated in .atomic_check */ 5300670c2f5SKerem Karabay appletbdrm_state->request = NULL; 5310670c2f5SKerem Karabay appletbdrm_state->response = NULL; 5320670c2f5SKerem Karabay 5330670c2f5SKerem Karabay appletbdrm_state->request_size = 0; 5340670c2f5SKerem Karabay appletbdrm_state->frames_size = 0; 5350670c2f5SKerem Karabay 5360670c2f5SKerem Karabay new_shadow_plane_state = &appletbdrm_state->base; 5370670c2f5SKerem Karabay 5380670c2f5SKerem Karabay __drm_gem_duplicate_shadow_plane_state(plane, new_shadow_plane_state); 5390670c2f5SKerem Karabay 5400670c2f5SKerem Karabay return &new_shadow_plane_state->base; 5410670c2f5SKerem Karabay } 5420670c2f5SKerem Karabay 5430670c2f5SKerem Karabay static void appletbdrm_primary_plane_destroy_state(struct drm_plane *plane, 5440670c2f5SKerem Karabay struct drm_plane_state *state) 5450670c2f5SKerem Karabay { 5460670c2f5SKerem Karabay struct appletbdrm_plane_state *appletbdrm_state = to_appletbdrm_plane_state(state); 5470670c2f5SKerem Karabay 5480670c2f5SKerem Karabay kfree(appletbdrm_state->request); 5490670c2f5SKerem Karabay kfree(appletbdrm_state->response); 5500670c2f5SKerem Karabay 5510670c2f5SKerem Karabay __drm_gem_destroy_shadow_plane_state(&appletbdrm_state->base); 5520670c2f5SKerem Karabay 5530670c2f5SKerem Karabay kfree(appletbdrm_state); 5540670c2f5SKerem Karabay } 5550670c2f5SKerem Karabay 5560670c2f5SKerem Karabay static const struct drm_plane_helper_funcs appletbdrm_primary_plane_helper_funcs = { 5570670c2f5SKerem Karabay DRM_GEM_SHADOW_PLANE_HELPER_FUNCS, 5580670c2f5SKerem Karabay .atomic_check = appletbdrm_primary_plane_helper_atomic_check, 5590670c2f5SKerem Karabay .atomic_update = appletbdrm_primary_plane_helper_atomic_update, 5600670c2f5SKerem Karabay .atomic_disable = appletbdrm_primary_plane_helper_atomic_disable, 5610670c2f5SKerem Karabay }; 5620670c2f5SKerem Karabay 5630670c2f5SKerem Karabay static const struct drm_plane_funcs appletbdrm_primary_plane_funcs = { 5640670c2f5SKerem Karabay .update_plane = drm_atomic_helper_update_plane, 5650670c2f5SKerem Karabay .disable_plane = drm_atomic_helper_disable_plane, 5660670c2f5SKerem Karabay .reset = appletbdrm_primary_plane_reset, 5670670c2f5SKerem Karabay .atomic_duplicate_state = appletbdrm_primary_plane_duplicate_state, 5680670c2f5SKerem Karabay .atomic_destroy_state = appletbdrm_primary_plane_destroy_state, 5690670c2f5SKerem Karabay .destroy = drm_plane_cleanup, 5700670c2f5SKerem Karabay }; 5710670c2f5SKerem Karabay 5720670c2f5SKerem Karabay static enum drm_mode_status appletbdrm_crtc_helper_mode_valid(struct drm_crtc *crtc, 5730670c2f5SKerem Karabay const struct drm_display_mode *mode) 5740670c2f5SKerem Karabay { 5750670c2f5SKerem Karabay struct appletbdrm_device *adev = drm_to_adev(crtc->dev); 5760670c2f5SKerem Karabay 5770670c2f5SKerem Karabay return drm_crtc_helper_mode_valid_fixed(crtc, mode, &adev->mode); 5780670c2f5SKerem Karabay } 5790670c2f5SKerem Karabay 5800670c2f5SKerem Karabay static const struct drm_mode_config_funcs appletbdrm_mode_config_funcs = { 5810670c2f5SKerem Karabay .fb_create = drm_gem_fb_create_with_dirty, 5820670c2f5SKerem Karabay .atomic_check = drm_atomic_helper_check, 5830670c2f5SKerem Karabay .atomic_commit = drm_atomic_helper_commit, 5840670c2f5SKerem Karabay }; 5850670c2f5SKerem Karabay 5860670c2f5SKerem Karabay static const struct drm_connector_funcs appletbdrm_connector_funcs = { 5870670c2f5SKerem Karabay .reset = drm_atomic_helper_connector_reset, 5880670c2f5SKerem Karabay .destroy = drm_connector_cleanup, 5890670c2f5SKerem Karabay .fill_modes = drm_helper_probe_single_connector_modes, 5900670c2f5SKerem Karabay .atomic_destroy_state = drm_atomic_helper_connector_destroy_state, 5910670c2f5SKerem Karabay .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, 5920670c2f5SKerem Karabay }; 5930670c2f5SKerem Karabay 5940670c2f5SKerem Karabay static const struct drm_connector_helper_funcs appletbdrm_connector_helper_funcs = { 5950670c2f5SKerem Karabay .get_modes = appletbdrm_connector_helper_get_modes, 5960670c2f5SKerem Karabay }; 5970670c2f5SKerem Karabay 5980670c2f5SKerem Karabay static const struct drm_crtc_helper_funcs appletbdrm_crtc_helper_funcs = { 5990670c2f5SKerem Karabay .mode_valid = appletbdrm_crtc_helper_mode_valid, 6000670c2f5SKerem Karabay }; 6010670c2f5SKerem Karabay 6020670c2f5SKerem Karabay static const struct drm_crtc_funcs appletbdrm_crtc_funcs = { 6030670c2f5SKerem Karabay .reset = drm_atomic_helper_crtc_reset, 6040670c2f5SKerem Karabay .destroy = drm_crtc_cleanup, 6050670c2f5SKerem Karabay .set_config = drm_atomic_helper_set_config, 6060670c2f5SKerem Karabay .page_flip = drm_atomic_helper_page_flip, 6070670c2f5SKerem Karabay .atomic_duplicate_state = drm_atomic_helper_crtc_duplicate_state, 6080670c2f5SKerem Karabay .atomic_destroy_state = drm_atomic_helper_crtc_destroy_state, 6090670c2f5SKerem Karabay }; 6100670c2f5SKerem Karabay 6110670c2f5SKerem Karabay static const struct drm_encoder_funcs appletbdrm_encoder_funcs = { 6120670c2f5SKerem Karabay .destroy = drm_encoder_cleanup, 6130670c2f5SKerem Karabay }; 6140670c2f5SKerem Karabay 6150670c2f5SKerem Karabay static struct drm_gem_object *appletbdrm_driver_gem_prime_import(struct drm_device *dev, 6160670c2f5SKerem Karabay struct dma_buf *dma_buf) 6170670c2f5SKerem Karabay { 6180670c2f5SKerem Karabay struct appletbdrm_device *adev = drm_to_adev(dev); 6190670c2f5SKerem Karabay 6200670c2f5SKerem Karabay if (!adev->dmadev) 6210670c2f5SKerem Karabay return ERR_PTR(-ENODEV); 6220670c2f5SKerem Karabay 6230670c2f5SKerem Karabay return drm_gem_prime_import_dev(dev, dma_buf, adev->dmadev); 6240670c2f5SKerem Karabay } 6250670c2f5SKerem Karabay 6260670c2f5SKerem Karabay DEFINE_DRM_GEM_FOPS(appletbdrm_drm_fops); 6270670c2f5SKerem Karabay 6280670c2f5SKerem Karabay static const struct drm_driver appletbdrm_drm_driver = { 6290670c2f5SKerem Karabay DRM_GEM_SHMEM_DRIVER_OPS, 6300670c2f5SKerem Karabay .gem_prime_import = appletbdrm_driver_gem_prime_import, 6310670c2f5SKerem Karabay .name = "appletbdrm", 6320670c2f5SKerem Karabay .desc = "Apple Touch Bar DRM Driver", 6330670c2f5SKerem Karabay .major = 1, 6340670c2f5SKerem Karabay .minor = 0, 6350670c2f5SKerem Karabay .driver_features = DRIVER_MODESET | DRIVER_GEM | DRIVER_ATOMIC, 6360670c2f5SKerem Karabay .fops = &appletbdrm_drm_fops, 6370670c2f5SKerem Karabay }; 6380670c2f5SKerem Karabay 6390670c2f5SKerem Karabay static int appletbdrm_setup_mode_config(struct appletbdrm_device *adev) 6400670c2f5SKerem Karabay { 6410670c2f5SKerem Karabay struct drm_connector *connector = &adev->connector; 6420670c2f5SKerem Karabay struct drm_plane *primary_plane; 6430670c2f5SKerem Karabay struct drm_crtc *crtc; 6440670c2f5SKerem Karabay struct drm_encoder *encoder; 6450670c2f5SKerem Karabay struct drm_device *drm = &adev->drm; 6460670c2f5SKerem Karabay int ret; 6470670c2f5SKerem Karabay 6480670c2f5SKerem Karabay ret = drmm_mode_config_init(drm); 6490670c2f5SKerem Karabay if (ret) { 6500670c2f5SKerem Karabay drm_err(drm, "Failed to initialize mode configuration\n"); 6510670c2f5SKerem Karabay return ret; 6520670c2f5SKerem Karabay } 6530670c2f5SKerem Karabay 6540670c2f5SKerem Karabay primary_plane = &adev->primary_plane; 6550670c2f5SKerem Karabay ret = drm_universal_plane_init(drm, primary_plane, 0, 6560670c2f5SKerem Karabay &appletbdrm_primary_plane_funcs, 6570670c2f5SKerem Karabay appletbdrm_primary_plane_formats, 6580670c2f5SKerem Karabay ARRAY_SIZE(appletbdrm_primary_plane_formats), 6590670c2f5SKerem Karabay NULL, 6600670c2f5SKerem Karabay DRM_PLANE_TYPE_PRIMARY, NULL); 6610670c2f5SKerem Karabay if (ret) { 6620670c2f5SKerem Karabay drm_err(drm, "Failed to initialize universal plane object\n"); 6630670c2f5SKerem Karabay return ret; 6640670c2f5SKerem Karabay } 6650670c2f5SKerem Karabay 6660670c2f5SKerem Karabay drm_plane_helper_add(primary_plane, &appletbdrm_primary_plane_helper_funcs); 6670670c2f5SKerem Karabay drm_plane_enable_fb_damage_clips(primary_plane); 6680670c2f5SKerem Karabay 6690670c2f5SKerem Karabay crtc = &adev->crtc; 6700670c2f5SKerem Karabay ret = drm_crtc_init_with_planes(drm, crtc, primary_plane, NULL, 6710670c2f5SKerem Karabay &appletbdrm_crtc_funcs, NULL); 6720670c2f5SKerem Karabay if (ret) { 6730670c2f5SKerem Karabay drm_err(drm, "Failed to initialize CRTC object\n"); 6740670c2f5SKerem Karabay return ret; 6750670c2f5SKerem Karabay } 6760670c2f5SKerem Karabay 6770670c2f5SKerem Karabay drm_crtc_helper_add(crtc, &appletbdrm_crtc_helper_funcs); 6780670c2f5SKerem Karabay 6790670c2f5SKerem Karabay encoder = &adev->encoder; 6800670c2f5SKerem Karabay ret = drm_encoder_init(drm, encoder, &appletbdrm_encoder_funcs, 6810670c2f5SKerem Karabay DRM_MODE_ENCODER_DAC, NULL); 6820670c2f5SKerem Karabay if (ret) { 6830670c2f5SKerem Karabay drm_err(drm, "Failed to initialize encoder\n"); 6840670c2f5SKerem Karabay return ret; 6850670c2f5SKerem Karabay } 6860670c2f5SKerem Karabay 6870670c2f5SKerem Karabay encoder->possible_crtcs = drm_crtc_mask(crtc); 6880670c2f5SKerem Karabay 6890670c2f5SKerem Karabay /* 6900670c2f5SKerem Karabay * The coordinate system used by the device is different from the 6910670c2f5SKerem Karabay * coordinate system of the framebuffer in that the x and y axes are 6920670c2f5SKerem Karabay * swapped, and that the y axis is inverted; so what the device reports 6930670c2f5SKerem Karabay * as the height is actually the width of the framebuffer and vice 6940670c2f5SKerem Karabay * versa. 6950670c2f5SKerem Karabay */ 6960670c2f5SKerem Karabay drm->mode_config.max_width = max(adev->height, DRM_SHADOW_PLANE_MAX_WIDTH); 6970670c2f5SKerem Karabay drm->mode_config.max_height = max(adev->width, DRM_SHADOW_PLANE_MAX_HEIGHT); 6980670c2f5SKerem Karabay drm->mode_config.preferred_depth = APPLETBDRM_BITS_PER_PIXEL; 6990670c2f5SKerem Karabay drm->mode_config.funcs = &appletbdrm_mode_config_funcs; 7000670c2f5SKerem Karabay 7010670c2f5SKerem Karabay adev->mode = (struct drm_display_mode) { 7020670c2f5SKerem Karabay DRM_MODE_INIT(60, adev->height, adev->width, 7030670c2f5SKerem Karabay DRM_MODE_RES_MM(adev->height, 218), 7040670c2f5SKerem Karabay DRM_MODE_RES_MM(adev->width, 218)) 7050670c2f5SKerem Karabay }; 7060670c2f5SKerem Karabay 7070670c2f5SKerem Karabay ret = drm_connector_init(drm, connector, 7080670c2f5SKerem Karabay &appletbdrm_connector_funcs, DRM_MODE_CONNECTOR_USB); 7090670c2f5SKerem Karabay if (ret) { 7100670c2f5SKerem Karabay drm_err(drm, "Failed to initialize connector\n"); 7110670c2f5SKerem Karabay return ret; 7120670c2f5SKerem Karabay } 7130670c2f5SKerem Karabay 7140670c2f5SKerem Karabay drm_connector_helper_add(connector, &appletbdrm_connector_helper_funcs); 7150670c2f5SKerem Karabay 7160670c2f5SKerem Karabay ret = drm_connector_set_panel_orientation(connector, 7170670c2f5SKerem Karabay DRM_MODE_PANEL_ORIENTATION_RIGHT_UP); 7180670c2f5SKerem Karabay if (ret) { 7190670c2f5SKerem Karabay drm_err(drm, "Failed to set panel orientation\n"); 7200670c2f5SKerem Karabay return ret; 7210670c2f5SKerem Karabay } 7220670c2f5SKerem Karabay 7230670c2f5SKerem Karabay connector->display_info.non_desktop = true; 7240670c2f5SKerem Karabay ret = drm_object_property_set_value(&connector->base, 7250670c2f5SKerem Karabay drm->mode_config.non_desktop_property, true); 7260670c2f5SKerem Karabay if (ret) { 7270670c2f5SKerem Karabay drm_err(drm, "Failed to set non-desktop property\n"); 7280670c2f5SKerem Karabay return ret; 7290670c2f5SKerem Karabay } 7300670c2f5SKerem Karabay 7310670c2f5SKerem Karabay ret = drm_connector_attach_encoder(connector, encoder); 7320670c2f5SKerem Karabay 7330670c2f5SKerem Karabay if (ret) { 7340670c2f5SKerem Karabay drm_err(drm, "Failed to initialize simple display pipe\n"); 7350670c2f5SKerem Karabay return ret; 7360670c2f5SKerem Karabay } 7370670c2f5SKerem Karabay 7380670c2f5SKerem Karabay drm_mode_config_reset(drm); 7390670c2f5SKerem Karabay 7400670c2f5SKerem Karabay return 0; 7410670c2f5SKerem Karabay } 7420670c2f5SKerem Karabay 7430670c2f5SKerem Karabay static int appletbdrm_probe(struct usb_interface *intf, 7440670c2f5SKerem Karabay const struct usb_device_id *id) 7450670c2f5SKerem Karabay { 7460670c2f5SKerem Karabay struct usb_endpoint_descriptor *bulk_in, *bulk_out; 7470670c2f5SKerem Karabay struct device *dev = &intf->dev; 7480670c2f5SKerem Karabay struct appletbdrm_device *adev; 7490670c2f5SKerem Karabay struct drm_device *drm = NULL; 7500670c2f5SKerem Karabay int ret; 7510670c2f5SKerem Karabay 7520670c2f5SKerem Karabay ret = usb_find_common_endpoints(intf->cur_altsetting, &bulk_in, &bulk_out, NULL, NULL); 7530670c2f5SKerem Karabay if (ret) { 7540670c2f5SKerem Karabay drm_err(drm, "appletbdrm: Failed to find bulk endpoints\n"); 7550670c2f5SKerem Karabay return ret; 7560670c2f5SKerem Karabay } 7570670c2f5SKerem Karabay 7580670c2f5SKerem Karabay adev = devm_drm_dev_alloc(dev, &appletbdrm_drm_driver, struct appletbdrm_device, drm); 7590670c2f5SKerem Karabay if (IS_ERR(adev)) 7600670c2f5SKerem Karabay return PTR_ERR(adev); 7610670c2f5SKerem Karabay 7620670c2f5SKerem Karabay adev->in_ep = bulk_in->bEndpointAddress; 7630670c2f5SKerem Karabay adev->out_ep = bulk_out->bEndpointAddress; 7640670c2f5SKerem Karabay adev->dmadev = dev; 7650670c2f5SKerem Karabay 7660670c2f5SKerem Karabay drm = &adev->drm; 7670670c2f5SKerem Karabay 7680670c2f5SKerem Karabay usb_set_intfdata(intf, adev); 7690670c2f5SKerem Karabay 7700670c2f5SKerem Karabay ret = appletbdrm_get_information(adev); 7710670c2f5SKerem Karabay if (ret) { 7720670c2f5SKerem Karabay drm_err(drm, "Failed to get display information\n"); 7730670c2f5SKerem Karabay return ret; 7740670c2f5SKerem Karabay } 7750670c2f5SKerem Karabay 7760670c2f5SKerem Karabay ret = appletbdrm_signal_readiness(adev); 7770670c2f5SKerem Karabay if (ret) { 7780670c2f5SKerem Karabay drm_err(drm, "Failed to signal readiness\n"); 7790670c2f5SKerem Karabay return ret; 7800670c2f5SKerem Karabay } 7810670c2f5SKerem Karabay 7820670c2f5SKerem Karabay ret = appletbdrm_setup_mode_config(adev); 7830670c2f5SKerem Karabay if (ret) { 7840670c2f5SKerem Karabay drm_err(drm, "Failed to setup mode config\n"); 7850670c2f5SKerem Karabay return ret; 7860670c2f5SKerem Karabay } 7870670c2f5SKerem Karabay 7880670c2f5SKerem Karabay ret = drm_dev_register(drm, 0); 7890670c2f5SKerem Karabay if (ret) { 7900670c2f5SKerem Karabay drm_err(drm, "Failed to register DRM device\n"); 7910670c2f5SKerem Karabay return ret; 7920670c2f5SKerem Karabay } 7930670c2f5SKerem Karabay 7940670c2f5SKerem Karabay ret = appletbdrm_clear_display(adev); 7950670c2f5SKerem Karabay if (ret) { 7960670c2f5SKerem Karabay drm_err(drm, "Failed to clear display\n"); 7970670c2f5SKerem Karabay return ret; 7980670c2f5SKerem Karabay } 7990670c2f5SKerem Karabay 8000670c2f5SKerem Karabay return 0; 8010670c2f5SKerem Karabay } 8020670c2f5SKerem Karabay 8030670c2f5SKerem Karabay static void appletbdrm_disconnect(struct usb_interface *intf) 8040670c2f5SKerem Karabay { 8050670c2f5SKerem Karabay struct appletbdrm_device *adev = usb_get_intfdata(intf); 8060670c2f5SKerem Karabay struct drm_device *drm = &adev->drm; 8070670c2f5SKerem Karabay 8080670c2f5SKerem Karabay drm_dev_unplug(drm); 8090670c2f5SKerem Karabay drm_atomic_helper_shutdown(drm); 8100670c2f5SKerem Karabay } 8110670c2f5SKerem Karabay 8120670c2f5SKerem Karabay static void appletbdrm_shutdown(struct usb_interface *intf) 8130670c2f5SKerem Karabay { 8140670c2f5SKerem Karabay struct appletbdrm_device *adev = usb_get_intfdata(intf); 8150670c2f5SKerem Karabay 8160670c2f5SKerem Karabay /* 8170670c2f5SKerem Karabay * The framebuffer needs to be cleared on shutdown since its content 8180670c2f5SKerem Karabay * persists across boots 8190670c2f5SKerem Karabay */ 8200670c2f5SKerem Karabay drm_atomic_helper_shutdown(&adev->drm); 8210670c2f5SKerem Karabay } 8220670c2f5SKerem Karabay 8230670c2f5SKerem Karabay static const struct usb_device_id appletbdrm_usb_id_table[] = { 8240670c2f5SKerem Karabay { USB_DEVICE_INTERFACE_CLASS(0x05ac, 0x8302, USB_CLASS_AUDIO_VIDEO) }, 8250670c2f5SKerem Karabay {} 8260670c2f5SKerem Karabay }; 8270670c2f5SKerem Karabay MODULE_DEVICE_TABLE(usb, appletbdrm_usb_id_table); 8280670c2f5SKerem Karabay 8290670c2f5SKerem Karabay static struct usb_driver appletbdrm_usb_driver = { 8300670c2f5SKerem Karabay .name = "appletbdrm", 8310670c2f5SKerem Karabay .probe = appletbdrm_probe, 8320670c2f5SKerem Karabay .disconnect = appletbdrm_disconnect, 8330670c2f5SKerem Karabay .shutdown = appletbdrm_shutdown, 8340670c2f5SKerem Karabay .id_table = appletbdrm_usb_id_table, 8350670c2f5SKerem Karabay }; 8360670c2f5SKerem Karabay module_usb_driver(appletbdrm_usb_driver); 8370670c2f5SKerem Karabay 8380670c2f5SKerem Karabay MODULE_AUTHOR("Kerem Karabay <kekrby@gmail.com>"); 8390670c2f5SKerem Karabay MODULE_DESCRIPTION("Apple Touch Bar DRM Driver"); 8400670c2f5SKerem Karabay MODULE_LICENSE("GPL"); 841