xref: /openbmc/linux/drivers/gpu/drm/vboxvideo/hgsmi_base.c (revision e50e86dbcabda570fc8a1435fe2fca97e9ab7312)
1131abc56SHans de Goede // SPDX-License-Identifier: MIT
2131abc56SHans de Goede /* Copyright (C) 2006-2017 Oracle Corporation */
3131abc56SHans de Goede 
4131abc56SHans de Goede #include <linux/vbox_err.h>
5131abc56SHans de Goede #include "vbox_drv.h"
6131abc56SHans de Goede #include "vboxvideo_guest.h"
7131abc56SHans de Goede #include "vboxvideo_vbe.h"
8131abc56SHans de Goede #include "hgsmi_channels.h"
9131abc56SHans de Goede #include "hgsmi_ch_setup.h"
10131abc56SHans de Goede 
11131abc56SHans de Goede /**
12a3dd6d90SLee Jones  * hgsmi_report_flags_location - Inform the host of the location of
13a3dd6d90SLee Jones  *                               the host flags in VRAM via an HGSMI cmd.
14131abc56SHans de Goede  * Return: 0 or negative errno value.
15131abc56SHans de Goede  * @ctx:        The context of the guest heap to use.
16131abc56SHans de Goede  * @location:   The offset chosen for the flags within guest VRAM.
17131abc56SHans de Goede  */
hgsmi_report_flags_location(struct gen_pool * ctx,u32 location)18131abc56SHans de Goede int hgsmi_report_flags_location(struct gen_pool *ctx, u32 location)
19131abc56SHans de Goede {
20131abc56SHans de Goede 	struct hgsmi_buffer_location *p;
21131abc56SHans de Goede 
22131abc56SHans de Goede 	p = hgsmi_buffer_alloc(ctx, sizeof(*p), HGSMI_CH_HGSMI,
23131abc56SHans de Goede 			       HGSMI_CC_HOST_FLAGS_LOCATION);
24131abc56SHans de Goede 	if (!p)
25131abc56SHans de Goede 		return -ENOMEM;
26131abc56SHans de Goede 
27131abc56SHans de Goede 	p->buf_location = location;
28131abc56SHans de Goede 	p->buf_len = sizeof(struct hgsmi_host_flags);
29131abc56SHans de Goede 
30131abc56SHans de Goede 	hgsmi_buffer_submit(ctx, p);
31131abc56SHans de Goede 	hgsmi_buffer_free(ctx, p);
32131abc56SHans de Goede 
33131abc56SHans de Goede 	return 0;
34131abc56SHans de Goede }
35131abc56SHans de Goede 
36131abc56SHans de Goede /**
37a3dd6d90SLee Jones  * hgsmi_send_caps_info - Notify the host of HGSMI-related guest capabilities
38a3dd6d90SLee Jones  *                        via an HGSMI command.
39131abc56SHans de Goede  * Return: 0 or negative errno value.
40131abc56SHans de Goede  * @ctx:        The context of the guest heap to use.
41131abc56SHans de Goede  * @caps:       The capabilities to report, see vbva_caps.
42131abc56SHans de Goede  */
hgsmi_send_caps_info(struct gen_pool * ctx,u32 caps)43131abc56SHans de Goede int hgsmi_send_caps_info(struct gen_pool *ctx, u32 caps)
44131abc56SHans de Goede {
45131abc56SHans de Goede 	struct vbva_caps *p;
46131abc56SHans de Goede 
47131abc56SHans de Goede 	p = hgsmi_buffer_alloc(ctx, sizeof(*p), HGSMI_CH_VBVA, VBVA_INFO_CAPS);
48131abc56SHans de Goede 	if (!p)
49131abc56SHans de Goede 		return -ENOMEM;
50131abc56SHans de Goede 
51131abc56SHans de Goede 	p->rc = VERR_NOT_IMPLEMENTED;
52131abc56SHans de Goede 	p->caps = caps;
53131abc56SHans de Goede 
54131abc56SHans de Goede 	hgsmi_buffer_submit(ctx, p);
55131abc56SHans de Goede 
56131abc56SHans de Goede 	WARN_ON_ONCE(p->rc < 0);
57131abc56SHans de Goede 
58131abc56SHans de Goede 	hgsmi_buffer_free(ctx, p);
59131abc56SHans de Goede 
60131abc56SHans de Goede 	return 0;
61131abc56SHans de Goede }
62131abc56SHans de Goede 
hgsmi_test_query_conf(struct gen_pool * ctx)63131abc56SHans de Goede int hgsmi_test_query_conf(struct gen_pool *ctx)
64131abc56SHans de Goede {
65131abc56SHans de Goede 	u32 value = 0;
66131abc56SHans de Goede 	int ret;
67131abc56SHans de Goede 
68131abc56SHans de Goede 	ret = hgsmi_query_conf(ctx, U32_MAX, &value);
69131abc56SHans de Goede 	if (ret)
70131abc56SHans de Goede 		return ret;
71131abc56SHans de Goede 
72131abc56SHans de Goede 	return value == U32_MAX ? 0 : -EIO;
73131abc56SHans de Goede }
74131abc56SHans de Goede 
75131abc56SHans de Goede /**
76a3dd6d90SLee Jones  * hgsmi_query_conf - Query the host for an HGSMI configuration
77a3dd6d90SLee Jones  *                    parameter via an HGSMI command.
78131abc56SHans de Goede  * Return: 0 or negative errno value.
79131abc56SHans de Goede  * @ctx:        The context containing the heap used.
80131abc56SHans de Goede  * @index:      The index of the parameter to query.
81131abc56SHans de Goede  * @value_ret:  Where to store the value of the parameter on success.
82131abc56SHans de Goede  */
hgsmi_query_conf(struct gen_pool * ctx,u32 index,u32 * value_ret)83131abc56SHans de Goede int hgsmi_query_conf(struct gen_pool *ctx, u32 index, u32 *value_ret)
84131abc56SHans de Goede {
85131abc56SHans de Goede 	struct vbva_conf32 *p;
86131abc56SHans de Goede 
87131abc56SHans de Goede 	p = hgsmi_buffer_alloc(ctx, sizeof(*p), HGSMI_CH_VBVA,
88131abc56SHans de Goede 			       VBVA_QUERY_CONF32);
89131abc56SHans de Goede 	if (!p)
90131abc56SHans de Goede 		return -ENOMEM;
91131abc56SHans de Goede 
92131abc56SHans de Goede 	p->index = index;
93131abc56SHans de Goede 	p->value = U32_MAX;
94131abc56SHans de Goede 
95131abc56SHans de Goede 	hgsmi_buffer_submit(ctx, p);
96131abc56SHans de Goede 
97131abc56SHans de Goede 	*value_ret = p->value;
98131abc56SHans de Goede 
99131abc56SHans de Goede 	hgsmi_buffer_free(ctx, p);
100131abc56SHans de Goede 
101131abc56SHans de Goede 	return 0;
102131abc56SHans de Goede }
103131abc56SHans de Goede 
104131abc56SHans de Goede /**
105a3dd6d90SLee Jones  * hgsmi_update_pointer_shape - Pass the host a new mouse pointer shape
106a3dd6d90SLee Jones  *                              via an HGSMI command.
107131abc56SHans de Goede  * Return: 0 or negative errno value.
108131abc56SHans de Goede  * @ctx:        The context containing the heap to be used.
109131abc56SHans de Goede  * @flags:      Cursor flags.
110131abc56SHans de Goede  * @hot_x:      Horizontal position of the hot spot.
111131abc56SHans de Goede  * @hot_y:      Vertical position of the hot spot.
112131abc56SHans de Goede  * @width:      Width in pixels of the cursor.
113131abc56SHans de Goede  * @height:     Height in pixels of the cursor.
114131abc56SHans de Goede  * @pixels:     Pixel data, @see VMMDevReqMousePointer for the format.
115131abc56SHans de Goede  * @len:        Size in bytes of the pixel data.
116131abc56SHans de Goede  */
hgsmi_update_pointer_shape(struct gen_pool * ctx,u32 flags,u32 hot_x,u32 hot_y,u32 width,u32 height,u8 * pixels,u32 len)117131abc56SHans de Goede int hgsmi_update_pointer_shape(struct gen_pool *ctx, u32 flags,
118131abc56SHans de Goede 			       u32 hot_x, u32 hot_y, u32 width, u32 height,
119131abc56SHans de Goede 			       u8 *pixels, u32 len)
120131abc56SHans de Goede {
121131abc56SHans de Goede 	struct vbva_mouse_pointer_shape *p;
122131abc56SHans de Goede 	u32 pixel_len = 0;
123131abc56SHans de Goede 	int rc;
124131abc56SHans de Goede 
125131abc56SHans de Goede 	if (flags & VBOX_MOUSE_POINTER_SHAPE) {
126131abc56SHans de Goede 		/*
127131abc56SHans de Goede 		 * Size of the pointer data:
128131abc56SHans de Goede 		 * sizeof (AND mask) + sizeof (XOR_MASK)
129131abc56SHans de Goede 		 */
130131abc56SHans de Goede 		pixel_len = ((((width + 7) / 8) * height + 3) & ~3) +
131131abc56SHans de Goede 			 width * 4 * height;
132131abc56SHans de Goede 		if (pixel_len > len)
133131abc56SHans de Goede 			return -EINVAL;
134131abc56SHans de Goede 
135131abc56SHans de Goede 		/*
136131abc56SHans de Goede 		 * If shape is supplied, then always create the pointer visible.
137131abc56SHans de Goede 		 * See comments in 'vboxUpdatePointerShape'
138131abc56SHans de Goede 		 */
139131abc56SHans de Goede 		flags |= VBOX_MOUSE_POINTER_VISIBLE;
140131abc56SHans de Goede 	}
141131abc56SHans de Goede 
142*9eb32bd2SHans de Goede 	/*
143*9eb32bd2SHans de Goede 	 * The 4 extra bytes come from switching struct vbva_mouse_pointer_shape
144*9eb32bd2SHans de Goede 	 * from having a 4 bytes fixed array at the end to using a proper VLA
145*9eb32bd2SHans de Goede 	 * at the end. These 4 extra bytes were not subtracted from sizeof(*p)
146*9eb32bd2SHans de Goede 	 * before the switch to the VLA, so this way the behavior is unchanged.
147*9eb32bd2SHans de Goede 	 * Chances are these 4 extra bytes are not necessary but they are kept
148*9eb32bd2SHans de Goede 	 * to avoid regressions.
149*9eb32bd2SHans de Goede 	 */
150*9eb32bd2SHans de Goede 	p = hgsmi_buffer_alloc(ctx, sizeof(*p) + pixel_len + 4, HGSMI_CH_VBVA,
151131abc56SHans de Goede 			       VBVA_MOUSE_POINTER_SHAPE);
152131abc56SHans de Goede 	if (!p)
153131abc56SHans de Goede 		return -ENOMEM;
154131abc56SHans de Goede 
155131abc56SHans de Goede 	p->result = VINF_SUCCESS;
156131abc56SHans de Goede 	p->flags = flags;
157131abc56SHans de Goede 	p->hot_X = hot_x;
158131abc56SHans de Goede 	p->hot_y = hot_y;
159131abc56SHans de Goede 	p->width = width;
160131abc56SHans de Goede 	p->height = height;
161131abc56SHans de Goede 	if (pixel_len)
162131abc56SHans de Goede 		memcpy(p->data, pixels, pixel_len);
163131abc56SHans de Goede 
164131abc56SHans de Goede 	hgsmi_buffer_submit(ctx, p);
165131abc56SHans de Goede 
166131abc56SHans de Goede 	switch (p->result) {
167131abc56SHans de Goede 	case VINF_SUCCESS:
168131abc56SHans de Goede 		rc = 0;
169131abc56SHans de Goede 		break;
170131abc56SHans de Goede 	case VERR_NO_MEMORY:
171131abc56SHans de Goede 		rc = -ENOMEM;
172131abc56SHans de Goede 		break;
173131abc56SHans de Goede 	case VERR_NOT_SUPPORTED:
174131abc56SHans de Goede 		rc = -EBUSY;
175131abc56SHans de Goede 		break;
176131abc56SHans de Goede 	default:
177131abc56SHans de Goede 		rc = -EINVAL;
178131abc56SHans de Goede 	}
179131abc56SHans de Goede 
180131abc56SHans de Goede 	hgsmi_buffer_free(ctx, p);
181131abc56SHans de Goede 
182131abc56SHans de Goede 	return rc;
183131abc56SHans de Goede }
184131abc56SHans de Goede 
185131abc56SHans de Goede /**
186a3dd6d90SLee Jones  * hgsmi_cursor_position - Report the guest cursor position.  The host may
187a3dd6d90SLee Jones  *                         wish to use this information to re-position its
188a3dd6d90SLee Jones  *                         own cursor (though this is currently unlikely).
189a3dd6d90SLee Jones  *                         The current host cursor position is returned.
190131abc56SHans de Goede  * Return: 0 or negative errno value.
191131abc56SHans de Goede  * @ctx:              The context containing the heap used.
192131abc56SHans de Goede  * @report_position:  Are we reporting a position?
193131abc56SHans de Goede  * @x:                Guest cursor X position.
194131abc56SHans de Goede  * @y:                Guest cursor Y position.
195131abc56SHans de Goede  * @x_host:           Host cursor X position is stored here.  Optional.
196131abc56SHans de Goede  * @y_host:           Host cursor Y position is stored here.  Optional.
197131abc56SHans de Goede  */
hgsmi_cursor_position(struct gen_pool * ctx,bool report_position,u32 x,u32 y,u32 * x_host,u32 * y_host)198131abc56SHans de Goede int hgsmi_cursor_position(struct gen_pool *ctx, bool report_position,
199131abc56SHans de Goede 			  u32 x, u32 y, u32 *x_host, u32 *y_host)
200131abc56SHans de Goede {
201131abc56SHans de Goede 	struct vbva_cursor_position *p;
202131abc56SHans de Goede 
203131abc56SHans de Goede 	p = hgsmi_buffer_alloc(ctx, sizeof(*p), HGSMI_CH_VBVA,
204131abc56SHans de Goede 			       VBVA_CURSOR_POSITION);
205131abc56SHans de Goede 	if (!p)
206131abc56SHans de Goede 		return -ENOMEM;
207131abc56SHans de Goede 
208131abc56SHans de Goede 	p->report_position = report_position;
209131abc56SHans de Goede 	p->x = x;
210131abc56SHans de Goede 	p->y = y;
211131abc56SHans de Goede 
212131abc56SHans de Goede 	hgsmi_buffer_submit(ctx, p);
213131abc56SHans de Goede 
214131abc56SHans de Goede 	*x_host = p->x;
215131abc56SHans de Goede 	*y_host = p->y;
216131abc56SHans de Goede 
217131abc56SHans de Goede 	hgsmi_buffer_free(ctx, p);
218131abc56SHans de Goede 
219131abc56SHans de Goede 	return 0;
220131abc56SHans de Goede }
221