2019-02-06 21:01:16 +07:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
|
|
|
|
2018-05-14 21:33:46 +07:00
|
|
|
#ifndef _VKMS_DRV_H_
|
|
|
|
#define _VKMS_DRV_H_
|
|
|
|
|
2018-05-17 06:56:21 +07:00
|
|
|
#include <drm/drmP.h>
|
|
|
|
#include <drm/drm.h>
|
2018-07-12 09:01:47 +07:00
|
|
|
#include <drm/drm_gem.h>
|
2018-05-17 06:56:21 +07:00
|
|
|
#include <drm/drm_encoder.h>
|
2018-07-12 09:02:26 +07:00
|
|
|
#include <linux/hrtimer.h>
|
2018-05-17 06:56:21 +07:00
|
|
|
|
2018-09-06 12:17:16 +07:00
|
|
|
#define XRES_MIN 20
|
|
|
|
#define YRES_MIN 20
|
2018-07-12 09:02:14 +07:00
|
|
|
|
|
|
|
#define XRES_DEF 1024
|
|
|
|
#define YRES_DEF 768
|
|
|
|
|
|
|
|
#define XRES_MAX 8192
|
|
|
|
#define YRES_MAX 8192
|
|
|
|
|
2018-09-06 12:19:11 +07:00
|
|
|
extern bool enable_cursor;
|
|
|
|
|
2018-08-02 08:10:26 +07:00
|
|
|
struct vkms_crc_data {
|
|
|
|
struct drm_framebuffer fb;
|
2018-09-06 12:18:26 +07:00
|
|
|
struct drm_rect src, dst;
|
|
|
|
unsigned int offset;
|
|
|
|
unsigned int pitch;
|
|
|
|
unsigned int cpp;
|
2018-08-02 08:10:26 +07:00
|
|
|
};
|
|
|
|
|
2018-08-02 08:08:22 +07:00
|
|
|
/**
|
|
|
|
* vkms_plane_state - Driver specific plane state
|
|
|
|
* @base: base plane state
|
2018-08-02 08:10:26 +07:00
|
|
|
* @crc_data: data required for CRC computation
|
2018-08-02 08:08:22 +07:00
|
|
|
*/
|
|
|
|
struct vkms_plane_state {
|
|
|
|
struct drm_plane_state base;
|
2018-08-02 08:10:26 +07:00
|
|
|
struct vkms_crc_data *crc_data;
|
2018-08-02 08:08:22 +07:00
|
|
|
};
|
|
|
|
|
2018-07-24 23:31:05 +07:00
|
|
|
/**
|
|
|
|
* vkms_crtc_state - Driver specific CRTC state
|
|
|
|
* @base: base CRTC state
|
2018-08-02 08:10:26 +07:00
|
|
|
* @crc_work: work struct to compute and add CRC entries
|
2018-09-04 04:18:17 +07:00
|
|
|
* @n_frame_start: start frame number for computed CRC
|
|
|
|
* @n_frame_end: end frame number for computed CRC
|
2018-07-24 23:31:05 +07:00
|
|
|
*/
|
|
|
|
struct vkms_crtc_state {
|
|
|
|
struct drm_crtc_state base;
|
2018-08-02 08:10:26 +07:00
|
|
|
struct work_struct crc_work;
|
drm/vkms: Fix crc worker races
The issue we have is that the crc worker might fall behind. We've
tried to handle this by tracking both the earliest frame for which it
still needs to compute a crc, and the last one. Plus when the
crtc_state changes, we have a new work item, which are all run in
order due to the ordered workqueue we allocate for each vkms crtc.
Trouble is there's been a few small issues in the current code:
- we need to capture frame_end in the vblank hrtimer, not in the
worker. The worker might run much later, and then we generate a lot
of crc for which there's already a different worker queued up.
- frame number might be 0, so create a new crc_pending boolean to
track this without confusion.
- we need to atomically grab frame_start/end and clear it, so do that
all in one go. This is not going to create a new race, because if we
race with the hrtimer then our work will be re-run.
- only race that can happen is the following:
1. worker starts
2. hrtimer runs and updates frame_end
3. worker grabs frame_start/end, already reading the new frame_end,
and clears crc_pending
4. hrtimer calls queue_work()
5. worker completes
6. worker gets re-run, crc_pending is false
Explain this case a bit better by rewording the comment.
v2: Demote warning level output to debug when we fail to requeue, this
is expected under high load when the crc worker can't quite keep up.
Cc: Shayenne Moura <shayenneluzmoura@gmail.com>
Cc: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Cc: Haneen Mohammed <hamohammed.sa@gmail.com>
Cc: Daniel Vetter <daniel@ffwll.ch>
Signed-off-by: Daniel Vetter <daniel.vetter@intel.com>
Reviewed-by: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Tested-by: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Signed-off-by: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190606222751.32567-2-daniel.vetter@ffwll.ch
2019-06-07 05:27:42 +07:00
|
|
|
|
2019-06-07 05:27:44 +07:00
|
|
|
/* below three are protected by vkms_output.crc_lock */
|
drm/vkms: Fix crc worker races
The issue we have is that the crc worker might fall behind. We've
tried to handle this by tracking both the earliest frame for which it
still needs to compute a crc, and the last one. Plus when the
crtc_state changes, we have a new work item, which are all run in
order due to the ordered workqueue we allocate for each vkms crtc.
Trouble is there's been a few small issues in the current code:
- we need to capture frame_end in the vblank hrtimer, not in the
worker. The worker might run much later, and then we generate a lot
of crc for which there's already a different worker queued up.
- frame number might be 0, so create a new crc_pending boolean to
track this without confusion.
- we need to atomically grab frame_start/end and clear it, so do that
all in one go. This is not going to create a new race, because if we
race with the hrtimer then our work will be re-run.
- only race that can happen is the following:
1. worker starts
2. hrtimer runs and updates frame_end
3. worker grabs frame_start/end, already reading the new frame_end,
and clears crc_pending
4. hrtimer calls queue_work()
5. worker completes
6. worker gets re-run, crc_pending is false
Explain this case a bit better by rewording the comment.
v2: Demote warning level output to debug when we fail to requeue, this
is expected under high load when the crc worker can't quite keep up.
Cc: Shayenne Moura <shayenneluzmoura@gmail.com>
Cc: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Cc: Haneen Mohammed <hamohammed.sa@gmail.com>
Cc: Daniel Vetter <daniel@ffwll.ch>
Signed-off-by: Daniel Vetter <daniel.vetter@intel.com>
Reviewed-by: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Tested-by: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Signed-off-by: Rodrigo Siqueira <rodrigosiqueiramelo@gmail.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190606222751.32567-2-daniel.vetter@ffwll.ch
2019-06-07 05:27:42 +07:00
|
|
|
bool crc_pending;
|
2018-09-04 04:18:17 +07:00
|
|
|
u64 frame_start;
|
|
|
|
u64 frame_end;
|
2018-07-24 23:31:05 +07:00
|
|
|
};
|
|
|
|
|
2018-05-17 06:56:21 +07:00
|
|
|
struct vkms_output {
|
|
|
|
struct drm_crtc crtc;
|
|
|
|
struct drm_encoder encoder;
|
|
|
|
struct drm_connector connector;
|
2018-07-12 09:02:26 +07:00
|
|
|
struct hrtimer vblank_hrtimer;
|
|
|
|
ktime_t period_ns;
|
|
|
|
struct drm_pending_vblank_event *event;
|
2018-08-02 08:10:26 +07:00
|
|
|
bool crc_enabled;
|
|
|
|
/* ordered wq for crc_work */
|
|
|
|
struct workqueue_struct *crc_workq;
|
|
|
|
/* protects concurrent access to crc_data */
|
|
|
|
spinlock_t lock;
|
2019-06-07 05:27:44 +07:00
|
|
|
|
|
|
|
spinlock_t crc_lock;
|
2018-05-17 06:56:21 +07:00
|
|
|
};
|
2018-05-14 21:33:46 +07:00
|
|
|
|
|
|
|
struct vkms_device {
|
|
|
|
struct drm_device drm;
|
|
|
|
struct platform_device *platform;
|
2018-05-17 06:56:21 +07:00
|
|
|
struct vkms_output output;
|
2018-05-14 21:33:46 +07:00
|
|
|
};
|
|
|
|
|
2018-07-12 09:01:47 +07:00
|
|
|
struct vkms_gem_object {
|
|
|
|
struct drm_gem_object gem;
|
|
|
|
struct mutex pages_lock; /* Page lock used in page fault handler */
|
|
|
|
struct page **pages;
|
2018-07-24 23:26:59 +07:00
|
|
|
unsigned int vmap_count;
|
|
|
|
void *vaddr;
|
2018-07-12 09:01:47 +07:00
|
|
|
};
|
|
|
|
|
2018-07-12 09:02:26 +07:00
|
|
|
#define drm_crtc_to_vkms_output(target) \
|
|
|
|
container_of(target, struct vkms_output, crtc)
|
|
|
|
|
|
|
|
#define drm_device_to_vkms_device(target) \
|
|
|
|
container_of(target, struct vkms_device, drm)
|
|
|
|
|
2018-07-24 23:26:59 +07:00
|
|
|
#define drm_gem_to_vkms_gem(target)\
|
|
|
|
container_of(target, struct vkms_gem_object, gem)
|
|
|
|
|
2018-07-24 23:31:05 +07:00
|
|
|
#define to_vkms_crtc_state(target)\
|
|
|
|
container_of(target, struct vkms_crtc_state, base)
|
|
|
|
|
2018-08-02 08:08:22 +07:00
|
|
|
#define to_vkms_plane_state(target)\
|
|
|
|
container_of(target, struct vkms_plane_state, base)
|
|
|
|
|
2018-07-12 09:02:26 +07:00
|
|
|
/* CRTC */
|
2018-05-17 06:56:21 +07:00
|
|
|
int vkms_crtc_init(struct drm_device *dev, struct drm_crtc *crtc,
|
|
|
|
struct drm_plane *primary, struct drm_plane *cursor);
|
|
|
|
|
2018-07-12 09:02:26 +07:00
|
|
|
bool vkms_get_vblank_timestamp(struct drm_device *dev, unsigned int pipe,
|
|
|
|
int *max_error, ktime_t *vblank_time,
|
|
|
|
bool in_vblank_irq);
|
|
|
|
|
2018-05-17 06:56:21 +07:00
|
|
|
int vkms_output_init(struct vkms_device *vkmsdev);
|
|
|
|
|
2018-09-06 12:17:16 +07:00
|
|
|
struct drm_plane *vkms_plane_init(struct vkms_device *vkmsdev,
|
|
|
|
enum drm_plane_type type);
|
2018-05-17 06:56:21 +07:00
|
|
|
|
2018-07-12 09:01:47 +07:00
|
|
|
/* Gem stuff */
|
|
|
|
struct drm_gem_object *vkms_gem_create(struct drm_device *dev,
|
|
|
|
struct drm_file *file,
|
|
|
|
u32 *handle,
|
|
|
|
u64 size);
|
|
|
|
|
2018-07-26 21:45:49 +07:00
|
|
|
vm_fault_t vkms_gem_fault(struct vm_fault *vmf);
|
2018-07-12 09:01:47 +07:00
|
|
|
|
|
|
|
int vkms_dumb_create(struct drm_file *file, struct drm_device *dev,
|
|
|
|
struct drm_mode_create_dumb *args);
|
|
|
|
|
|
|
|
void vkms_gem_free_object(struct drm_gem_object *obj);
|
|
|
|
|
2018-07-24 23:26:59 +07:00
|
|
|
int vkms_gem_vmap(struct drm_gem_object *obj);
|
|
|
|
|
|
|
|
void vkms_gem_vunmap(struct drm_gem_object *obj);
|
|
|
|
|
2018-08-02 08:10:26 +07:00
|
|
|
/* CRC Support */
|
2019-06-13 19:18:02 +07:00
|
|
|
const char *const *vkms_get_crc_sources(struct drm_crtc *crtc,
|
|
|
|
size_t *count);
|
2018-08-21 15:38:56 +07:00
|
|
|
int vkms_set_crc_source(struct drm_crtc *crtc, const char *src_name);
|
2018-08-21 15:38:55 +07:00
|
|
|
int vkms_verify_crc_source(struct drm_crtc *crtc, const char *source_name,
|
|
|
|
size_t *values_cnt);
|
2018-08-02 08:10:26 +07:00
|
|
|
void vkms_crc_work_handle(struct work_struct *work);
|
|
|
|
|
2018-05-14 21:33:46 +07:00
|
|
|
#endif /* _VKMS_DRV_H_ */
|