main: flush after execute_pending
[staging/windowmanager.git] / src / main.cpp
1 #include "util.h"
2 #include "wayland.hpp"
3
4 #include <unistd.h>
5
6 #include <stdlib.h>
7 #include <string.h>
8
9 #include <map>
10 #include <memory>
11 #include <string>
12 #include <vector>
13
14 #include <sys/poll.h>
15
16 struct conn {
17    std::vector<std::unique_ptr<wl::output>> outputs;
18    std::unique_ptr<genivi::controller> c;
19 };
20
21 namespace {
22 int check_events(struct wl::display *d, struct conn *c, int fd) {
23    struct pollfd pfd[2] = {{.fd = d->get_fd(), .events = POLLIN, .revents = 0},
24                            {.fd = fd, .events = POLLIN, .revents = 0}};
25
26    d->flush();
27
28    if (poll(pfd, fd != -1 ? 2 : 1, -1) != -1 && errno != EINTR) {
29       int ret = 0;
30
31       if (pfd[0].revents & POLLIN) {
32          ret = d->dispatch();
33       }
34
35       if (ret == -1)
36          return ret;
37
38       if (fd != -1 && (pfd[1].revents & POLLIN)) {
39          char buf[256];
40
41          // read all there is ...
42          while (read(pfd[1].fd, buf, sizeof(buf)) == sizeof(buf))
43             ;
44
45          // Display current status
46          if (!c->c->surfaces.empty()) {
47             puts("Surfaces:");
48             for (auto const &i : c->c->surfaces) {
49                struct genivi::rect const &r = i.second->dst_rect;
50                struct genivi::size const &s = i.second->size;
51                printf("%d [%ux%u] (%ux%u@%dx%d), ", i.first, s.w, s.h, r.w, r.h,
52                       r.x, r.y);
53             }
54             puts("\b\b ");
55          }
56
57          if (!c->c->layers.empty()) {
58             puts("Layers:");
59             for (auto const &i : c->c->layers) {
60                struct genivi::rect const &r = i.second->dst_rect;
61                struct genivi::size const &s = i.second->size;
62                printf("%d [%ux%u] (%ux%u@%dx%d), ", i.first, s.w, s.h, r.w, r.h,
63                       r.x, r.y);
64             }
65             puts("\b\b ");
66          }
67       }
68    }
69
70    return 0;
71 }
72
73 void init_layout(struct conn &c) {
74    auto &o = c.outputs.front();
75    auto &s = c.c->screens.begin()->second;
76    auto &layers = c.c->layers;
77
78    // XXX: Write output dimensions to ivi controller...
79    c.c->output_size = genivi::size{uint32_t(o->width), uint32_t(o->height)};
80
81    // Clear scene
82    layers.clear();
83
84    // Clear screen
85    s->clear();
86
87    // Setup our dummy scene...
88    c.c->layer_create(100, 0, 0);  // bottom layer, anything else
89    c.c->layer_create(1000, 0, 0); // top layer, mandelbrot
90
91    auto &l100 = c.c->layers[100];
92    auto &l1k = c.c->layers[1000];
93
94    // Set layers fullscreen
95    l100->set_destination_rectangle(0, 0, o->width, o->height);
96    l1k->set_destination_rectangle(0, 0, o->width, o->height);
97
98    // Add layers to screen
99    s->add_layer(l100.get());
100    s->add_layer(l1k.get());
101
102    c.c->commit_changes();
103    // Note: this does not flush the display!
104 }
105 }
106
107 int main(int argc, char **argv) {
108    lognotice("WinMan ver. %s", WINMAN_VERSION_STRING);
109
110    if (!getenv("XDG_RUNTIME_DIR"))
111       fatal("Environment variable XDG_RUNTIME_DIR not set");
112
113    auto d = std::make_unique<wl::display>();
114    if (!d->ok())
115       fatal("Could not connect to compositor");
116
117    struct conn c = {};
118
119    d->r->add_global_handler(
120       "ivi_controller", [&](wl_registry *r, uint32_t name, uint32_t v) {
121          c.c = std::make_unique<genivi::controller>(r, name, v);
122       });
123
124    d->r->add_global_handler(
125       "wl_output", [&](wl_registry *r, uint32_t name, uint32_t v) {
126          c.outputs.emplace_back(std::make_unique<wl::output>(r, name, v));
127       });
128
129    // First level objects
130    d->roundtrip();
131    // Second level objects
132    d->roundtrip();
133    // Third level objects
134    d->roundtrip();
135
136    if (!c.c)
137       fatal("ivi_controller global not available");
138
139    if (c.outputs.empty())
140       fatal("no output was set up!");
141
142    init_layout(c);
143
144    while (check_events(d.get(), &c, STDIN_FILENO) != -1) {
145       c.c->execute_pending();
146       d->flush();
147    }
148
149    d->roundtrip();
150
151    return 0;
152 }