main: also show layer info on line break
[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    struct wl::output &o = *c.outputs.begin()->get();
75    struct genivi::screen &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    // Setup our dummy scene...
82    if (layers.find(100) == layers.end()) {
83       logdebug("Creating layer 100 with output dimensions (%ux%u)", o.width, o.height);
84       c.c->layer_create(100, o.width, o.height);
85    }
86
87    struct genivi::layer &l = *c.c->layers[100].get();
88
89    l.set_destination_rectangle(0, 0, o.width, o.height);
90    s.clear();
91
92    logdebug("Add layer 100 to screen %u", s.id);
93    s.add_layer(&l);
94
95    c.c->commit_changes();
96    // Note: this does not flush the display!
97 }
98 }
99
100 int main(int argc, char **argv) {
101    lognotice("WinMan ver. %s", WINMAN_VERSION_STRING);
102
103    if (!getenv("XDG_RUNTIME_DIR"))
104       fatal("Environment variable XDG_RUNTIME_DIR not set");
105
106    auto d = std::make_unique<wl::display>();
107    if (!d->ok())
108       fatal("Could not connect to compositor");
109
110    struct conn c = {};
111
112    d->r->add_global_handler(
113       "ivi_controller", [&](wl_registry *r, uint32_t name, uint32_t v) {
114          c.c = std::make_unique<genivi::controller>(r, name, v);
115       });
116
117    d->r->add_global_handler(
118       "wl_output", [&](wl_registry *r, uint32_t name, uint32_t v) {
119          c.outputs.emplace_back(std::make_unique<wl::output>(r, name, v));
120       });
121
122    // First level objects
123    d->roundtrip();
124    // Second level objects
125    d->roundtrip();
126    // Third level objects
127    d->roundtrip();
128
129    if (!c.c)
130       fatal("ivi_controller global not available");
131
132    if (c.outputs.empty())
133       fatal("no output was set up!");
134
135    init_layout(c);
136
137    while (check_events(d.get(), &c, STDIN_FILENO) != -1) {
138       c.c->execute_pending();
139    }
140
141    d->roundtrip();
142
143    return 0;
144 }