blob: f89a24f50f2ebbb574d8a990f90c4870de2d2de9 [file] [log] [blame]
/**
* Copyright © 2016 IBM Corporation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <errno.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <systemd/sd-bus.h>
#include <systemd/sd-event.h>
/* Copied from phosphor-objmgr to make the code compile. */
static const int mapper_busy_retries = 5;
static const uint64_t mapper_busy_delay_interval_usec = 1000000;
static void quit(int r, void *loop)
{
sd_event_exit((sd_event *)loop, r);
}
static int callback(sd_bus_message *m, void *user, sd_bus_error *error)
{
(void) error;
sd_event *loop = user;
int r;
char *property = NULL;
r = sd_bus_message_skip(m, "s");
if (r < 0)
{
fprintf(stderr, "Error skipping message fields: %s\n",
strerror(-r));
quit(r, loop);
return r;
}
r = sd_bus_message_enter_container(m, SD_BUS_TYPE_ARRAY, "{sv}");
if (r < 0)
{
fprintf(stderr, "Error entering container: %s\n",
strerror(-r));
quit(r, loop);
return r;
}
while ((r = sd_bus_message_enter_container(
m,
SD_BUS_TYPE_DICT_ENTRY,
"sv")) > 0)
{
r = sd_bus_message_read(m, "s", &property);
if (r < 0)
{
fprintf(stderr, "Error reading message: %s\n",
strerror(-r));
quit(r, loop);
return r;
}
if (strcmp(property, "pgood"))
continue;
quit(0, loop);
break;
}
return 0;
}
static int get_object(sd_bus *conn, const char *obj,
sd_bus_message **reply)
{
sd_bus_message *request = NULL;
int r, retry = 0;
r = sd_bus_message_new_method_call(
conn, &request, "xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper", "GetObject");
if (r < 0)
goto exit;
r = sd_bus_message_append(request, "s", obj);
if (r < 0)
goto exit;
r = sd_bus_message_append(request, "as", 0, NULL);
if (r < 0)
goto exit;
while (true)
{
r = sd_bus_call(conn, request, 0, NULL, reply);
if (r == -EBUSY || r == -ENOBUFS)
{
if (retry >= mapper_busy_retries)
break;
usleep(mapper_busy_delay_interval_usec * (1 << retry));
++retry;
continue;
}
break;
}
if (r < 0)
goto exit;
exit:
sd_bus_message_unref(request);
return r;
}
static int get_service(sd_bus *conn, const char *obj, char **service)
{
sd_bus_message *reply = NULL;
const char *tmp;
int r;
r = get_object(conn, obj, &reply);
if (r < 0)
goto exit;
r = sd_bus_message_enter_container(reply, 0, NULL);
if (r < 0)
goto exit;
r = sd_bus_message_enter_container(reply, 0, NULL);
if (r < 0)
goto exit;
r = sd_bus_message_read(reply, "s", &tmp);
if (r < 0)
goto exit;
*service = strdup(tmp);
exit:
sd_bus_message_unref(reply);
return r;
}
int main(int argc, char *argv[])
{
static const char *matchfmt =
"type='signal',"
"interface='org.freedesktop.DBus.Properties',"
"member='PropertiesChanged',"
"arg0='org.openbmc.control.Power',"
"path='%s',"
"sender='%s'";
static const char *usage =
"Usage: %s OBJECTPATH on|off\n";
static const size_t LEN = 256;
sd_bus *conn = NULL;
sd_event *loop = NULL;
sd_bus_slot *slot = NULL;
sd_bus_error error = SD_BUS_ERROR_NULL;
char *service = NULL;
int r, dest = -1, state;
char match[LEN];
if (argc < 3)
{
fprintf(stderr, usage, argv[0]);
exit(EXIT_FAILURE);
}
if (!strcmp(argv[2], "on"))
dest = 1;
if (!strcmp(argv[2], "off"))
dest = 0;
if (dest != 0 && dest != 1)
{
fprintf(stderr, usage, argv[0]);
exit(EXIT_FAILURE);
}
r = sd_bus_default_system(&conn);
if (r < 0)
{
fprintf(stderr, "Error connecting to system bus: %s\n",
strerror(-r));
goto finish;
}
r = get_service(conn, argv[1], &service);
if (r < 0)
{
fprintf(stderr, "Error obtaining host service: %s\n",
strerror(-r));
goto finish;
}
r = sd_event_default(&loop);
if (r < 0)
{
fprintf(stderr, "Error obtaining event loop: %s\n",
strerror(-r));
goto finish;
}
r = sd_bus_attach_event(conn, loop, SD_EVENT_PRIORITY_NORMAL);
if (r < 0)
{
fprintf(stderr, "Failed to attach system "
"bus to event loop: %s\n",
strerror(-r));
goto finish;
}
if (strlen(matchfmt) + strnlen(argv[1], LEN) > LEN)
{
r = -E2BIG;
fprintf(stderr, "Error adding match rule: %s\n",
strerror(-r));
goto finish;
}
sprintf(match, matchfmt, argv[1], service);
r = sd_bus_add_match(conn,
&slot,
match,
callback,
loop);
if (r < 0)
{
fprintf(stderr, "Error adding match rule: %s\n",
strerror(-r));
goto finish;
}
r = sd_bus_get_property_trivial(conn,
service,
argv[1],
"org.openbmc.control.Power",
"pgood",
&error,
'i',
&state);
if (r < 0)
{
fprintf(stderr, "Error getting property: %s\n",
strerror(-r));
goto finish;
}
if (dest == state)
goto finish;
r = sd_event_loop(loop);
if (r < 0)
{
fprintf(stderr, "Error starting event loop: %s\n",
strerror(-r));
goto finish;
}
finish:
exit(r < 0 ? EXIT_FAILURE : EXIT_SUCCESS);
}