1*f0687c8aSRaman Tenneti #include <cstdio>
2*f0687c8aSRaman Tenneti #include <iostream>
3*f0687c8aSRaman Tenneti #include <unistd.h>
4*f0687c8aSRaman Tenneti #include <fcntl.h>
5*f0687c8aSRaman Tenneti #include <cassert>
6*f0687c8aSRaman Tenneti
7*f0687c8aSRaman Tenneti #include <kms++/kms++.h>
8*f0687c8aSRaman Tenneti #include "helpers.h"
9*f0687c8aSRaman Tenneti
10*f0687c8aSRaman Tenneti using namespace std;
11*f0687c8aSRaman Tenneti
12*f0687c8aSRaman Tenneti namespace kms
13*f0687c8aSRaman Tenneti {
14*f0687c8aSRaman Tenneti struct CrtcPriv {
15*f0687c8aSRaman Tenneti drmModeCrtcPtr drm_crtc;
16*f0687c8aSRaman Tenneti };
17*f0687c8aSRaman Tenneti
Crtc(Card & card,uint32_t id,uint32_t idx)18*f0687c8aSRaman Tenneti Crtc::Crtc(Card& card, uint32_t id, uint32_t idx)
19*f0687c8aSRaman Tenneti : DrmPropObject(card, id, DRM_MODE_OBJECT_CRTC, idx)
20*f0687c8aSRaman Tenneti {
21*f0687c8aSRaman Tenneti m_priv = new CrtcPriv();
22*f0687c8aSRaman Tenneti m_priv->drm_crtc = drmModeGetCrtc(this->card().fd(), this->id());
23*f0687c8aSRaman Tenneti assert(m_priv->drm_crtc);
24*f0687c8aSRaman Tenneti }
25*f0687c8aSRaman Tenneti
~Crtc()26*f0687c8aSRaman Tenneti Crtc::~Crtc()
27*f0687c8aSRaman Tenneti {
28*f0687c8aSRaman Tenneti drmModeFreeCrtc(m_priv->drm_crtc);
29*f0687c8aSRaman Tenneti delete m_priv;
30*f0687c8aSRaman Tenneti }
31*f0687c8aSRaman Tenneti
refresh()32*f0687c8aSRaman Tenneti void Crtc::refresh()
33*f0687c8aSRaman Tenneti {
34*f0687c8aSRaman Tenneti drmModeFreeCrtc(m_priv->drm_crtc);
35*f0687c8aSRaman Tenneti
36*f0687c8aSRaman Tenneti m_priv->drm_crtc = drmModeGetCrtc(this->card().fd(), this->id());
37*f0687c8aSRaman Tenneti assert(m_priv->drm_crtc);
38*f0687c8aSRaman Tenneti }
39*f0687c8aSRaman Tenneti
setup()40*f0687c8aSRaman Tenneti void Crtc::setup()
41*f0687c8aSRaman Tenneti {
42*f0687c8aSRaman Tenneti for (Plane* plane : card().get_planes()) {
43*f0687c8aSRaman Tenneti if (plane->supports_crtc(this))
44*f0687c8aSRaman Tenneti m_possible_planes.push_back(plane);
45*f0687c8aSRaman Tenneti }
46*f0687c8aSRaman Tenneti }
47*f0687c8aSRaman Tenneti
restore_mode(Connector * conn)48*f0687c8aSRaman Tenneti void Crtc::restore_mode(Connector* conn)
49*f0687c8aSRaman Tenneti {
50*f0687c8aSRaman Tenneti auto c = m_priv->drm_crtc;
51*f0687c8aSRaman Tenneti
52*f0687c8aSRaman Tenneti uint32_t conns[] = { conn->id() };
53*f0687c8aSRaman Tenneti
54*f0687c8aSRaman Tenneti drmModeSetCrtc(card().fd(), id(), c->buffer_id,
55*f0687c8aSRaman Tenneti c->x, c->y,
56*f0687c8aSRaman Tenneti conns, 1, &c->mode);
57*f0687c8aSRaman Tenneti }
58*f0687c8aSRaman Tenneti
set_mode(Connector * conn,const Videomode & mode)59*f0687c8aSRaman Tenneti int Crtc::set_mode(Connector* conn, const Videomode& mode)
60*f0687c8aSRaman Tenneti {
61*f0687c8aSRaman Tenneti AtomicReq req(card());
62*f0687c8aSRaman Tenneti
63*f0687c8aSRaman Tenneti unique_ptr<Blob> blob = mode.to_blob(card());
64*f0687c8aSRaman Tenneti
65*f0687c8aSRaman Tenneti req.add(conn, {
66*f0687c8aSRaman Tenneti { "CRTC_ID", this->id() },
67*f0687c8aSRaman Tenneti });
68*f0687c8aSRaman Tenneti
69*f0687c8aSRaman Tenneti req.add(this, {
70*f0687c8aSRaman Tenneti { "ACTIVE", 1 },
71*f0687c8aSRaman Tenneti { "MODE_ID", blob->id() },
72*f0687c8aSRaman Tenneti });
73*f0687c8aSRaman Tenneti
74*f0687c8aSRaman Tenneti int r = req.commit_sync(true);
75*f0687c8aSRaman Tenneti
76*f0687c8aSRaman Tenneti refresh();
77*f0687c8aSRaman Tenneti
78*f0687c8aSRaman Tenneti return r;
79*f0687c8aSRaman Tenneti }
80*f0687c8aSRaman Tenneti
set_mode(Connector * conn,Framebuffer & fb,const Videomode & mode)81*f0687c8aSRaman Tenneti int Crtc::set_mode(Connector* conn, Framebuffer& fb, const Videomode& mode)
82*f0687c8aSRaman Tenneti {
83*f0687c8aSRaman Tenneti uint32_t conns[] = { conn->id() };
84*f0687c8aSRaman Tenneti drmModeModeInfo drmmode = video_mode_to_drm_mode(mode);
85*f0687c8aSRaman Tenneti
86*f0687c8aSRaman Tenneti return drmModeSetCrtc(card().fd(), id(), fb.id(),
87*f0687c8aSRaman Tenneti 0, 0,
88*f0687c8aSRaman Tenneti conns, 1, &drmmode);
89*f0687c8aSRaman Tenneti }
90*f0687c8aSRaman Tenneti
disable_mode()91*f0687c8aSRaman Tenneti int Crtc::disable_mode()
92*f0687c8aSRaman Tenneti {
93*f0687c8aSRaman Tenneti return drmModeSetCrtc(card().fd(), id(), 0, 0, 0, 0, 0, 0);
94*f0687c8aSRaman Tenneti }
95*f0687c8aSRaman Tenneti
conv(float x)96*f0687c8aSRaman Tenneti static inline uint32_t conv(float x)
97*f0687c8aSRaman Tenneti {
98*f0687c8aSRaman Tenneti // XXX fix the conversion for fractional part
99*f0687c8aSRaman Tenneti return ((uint32_t)x) << 16;
100*f0687c8aSRaman Tenneti }
101*f0687c8aSRaman Tenneti
set_plane(Plane * plane,Framebuffer & fb,int32_t dst_x,int32_t dst_y,uint32_t dst_w,uint32_t dst_h,float src_x,float src_y,float src_w,float src_h)102*f0687c8aSRaman Tenneti int Crtc::set_plane(Plane* plane, Framebuffer& fb,
103*f0687c8aSRaman Tenneti int32_t dst_x, int32_t dst_y, uint32_t dst_w, uint32_t dst_h,
104*f0687c8aSRaman Tenneti float src_x, float src_y, float src_w, float src_h)
105*f0687c8aSRaman Tenneti {
106*f0687c8aSRaman Tenneti return drmModeSetPlane(card().fd(), plane->id(), id(), fb.id(), 0,
107*f0687c8aSRaman Tenneti dst_x, dst_y, dst_w, dst_h,
108*f0687c8aSRaman Tenneti conv(src_x), conv(src_y), conv(src_w), conv(src_h));
109*f0687c8aSRaman Tenneti }
110*f0687c8aSRaman Tenneti
disable_plane(Plane * plane)111*f0687c8aSRaman Tenneti int Crtc::disable_plane(Plane* plane)
112*f0687c8aSRaman Tenneti {
113*f0687c8aSRaman Tenneti return drmModeSetPlane(card().fd(), plane->id(), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
114*f0687c8aSRaman Tenneti }
115*f0687c8aSRaman Tenneti
get_primary_plane()116*f0687c8aSRaman Tenneti Plane* Crtc::get_primary_plane()
117*f0687c8aSRaman Tenneti {
118*f0687c8aSRaman Tenneti Plane* primary = nullptr;
119*f0687c8aSRaman Tenneti
120*f0687c8aSRaman Tenneti for (Plane* p : get_possible_planes()) {
121*f0687c8aSRaman Tenneti if (p->plane_type() != PlaneType::Primary)
122*f0687c8aSRaman Tenneti continue;
123*f0687c8aSRaman Tenneti
124*f0687c8aSRaman Tenneti if (p->crtc_id() == id())
125*f0687c8aSRaman Tenneti return p;
126*f0687c8aSRaman Tenneti
127*f0687c8aSRaman Tenneti primary = p;
128*f0687c8aSRaman Tenneti }
129*f0687c8aSRaman Tenneti
130*f0687c8aSRaman Tenneti if (primary)
131*f0687c8aSRaman Tenneti return primary;
132*f0687c8aSRaman Tenneti
133*f0687c8aSRaman Tenneti throw invalid_argument(string("No primary plane for crtc ") + to_string(id()));
134*f0687c8aSRaman Tenneti }
135*f0687c8aSRaman Tenneti
page_flip(Framebuffer & fb,void * data)136*f0687c8aSRaman Tenneti int Crtc::page_flip(Framebuffer& fb, void* data)
137*f0687c8aSRaman Tenneti {
138*f0687c8aSRaman Tenneti return drmModePageFlip(card().fd(), id(), fb.id(), DRM_MODE_PAGE_FLIP_EVENT, data);
139*f0687c8aSRaman Tenneti }
140*f0687c8aSRaman Tenneti
buffer_id() const141*f0687c8aSRaman Tenneti uint32_t Crtc::buffer_id() const
142*f0687c8aSRaman Tenneti {
143*f0687c8aSRaman Tenneti return m_priv->drm_crtc->buffer_id;
144*f0687c8aSRaman Tenneti }
145*f0687c8aSRaman Tenneti
x() const146*f0687c8aSRaman Tenneti uint32_t Crtc::x() const
147*f0687c8aSRaman Tenneti {
148*f0687c8aSRaman Tenneti return m_priv->drm_crtc->x;
149*f0687c8aSRaman Tenneti }
150*f0687c8aSRaman Tenneti
y() const151*f0687c8aSRaman Tenneti uint32_t Crtc::y() const
152*f0687c8aSRaman Tenneti {
153*f0687c8aSRaman Tenneti return m_priv->drm_crtc->y;
154*f0687c8aSRaman Tenneti }
155*f0687c8aSRaman Tenneti
width() const156*f0687c8aSRaman Tenneti uint32_t Crtc::width() const
157*f0687c8aSRaman Tenneti {
158*f0687c8aSRaman Tenneti return m_priv->drm_crtc->width;
159*f0687c8aSRaman Tenneti }
160*f0687c8aSRaman Tenneti
height() const161*f0687c8aSRaman Tenneti uint32_t Crtc::height() const
162*f0687c8aSRaman Tenneti {
163*f0687c8aSRaman Tenneti return m_priv->drm_crtc->height;
164*f0687c8aSRaman Tenneti }
165*f0687c8aSRaman Tenneti
mode_valid() const166*f0687c8aSRaman Tenneti int Crtc::mode_valid() const
167*f0687c8aSRaman Tenneti {
168*f0687c8aSRaman Tenneti return m_priv->drm_crtc->mode_valid;
169*f0687c8aSRaman Tenneti }
170*f0687c8aSRaman Tenneti
mode() const171*f0687c8aSRaman Tenneti Videomode Crtc::mode() const
172*f0687c8aSRaman Tenneti {
173*f0687c8aSRaman Tenneti return drm_mode_to_video_mode(m_priv->drm_crtc->mode);
174*f0687c8aSRaman Tenneti }
175*f0687c8aSRaman Tenneti
legacy_gamma_size() const176*f0687c8aSRaman Tenneti int Crtc::legacy_gamma_size() const
177*f0687c8aSRaman Tenneti {
178*f0687c8aSRaman Tenneti return m_priv->drm_crtc->gamma_size;
179*f0687c8aSRaman Tenneti }
180*f0687c8aSRaman Tenneti
legacy_gamma_set(vector<tuple<uint16_t,uint16_t,uint16_t>> v)181*f0687c8aSRaman Tenneti void Crtc::legacy_gamma_set(vector<tuple<uint16_t, uint16_t, uint16_t>> v)
182*f0687c8aSRaman Tenneti {
183*f0687c8aSRaman Tenneti uint32_t len = v.size();
184*f0687c8aSRaman Tenneti uint16_t red[len];
185*f0687c8aSRaman Tenneti uint16_t green[len];
186*f0687c8aSRaman Tenneti uint16_t blue[len];
187*f0687c8aSRaman Tenneti
188*f0687c8aSRaman Tenneti for (uint32_t i = 0; i < len; ++i) {
189*f0687c8aSRaman Tenneti red[i] = get<0>(v[i]);
190*f0687c8aSRaman Tenneti green[i] = get<1>(v[i]);
191*f0687c8aSRaman Tenneti blue[i] = get<2>(v[i]);
192*f0687c8aSRaman Tenneti }
193*f0687c8aSRaman Tenneti
194*f0687c8aSRaman Tenneti drmModeCrtcSetGamma(card().fd(), id(), len, red, green, blue);
195*f0687c8aSRaman Tenneti }
196*f0687c8aSRaman Tenneti
197*f0687c8aSRaman Tenneti } // namespace kms
198