2
0
mirror of https://github.com/openvswitch/ovs synced 2025-08-22 18:07:40 +00:00
ovs/lib/unixctl.c
Ethan Jackson ef8a3d1497 socket-util: Remove DSCP_INVALID.
The DSCP_INVALID flag allowed callers to prevent socket-util from
modify the DSCP bits of newly created sockets.  However, the two
really important callers (implementations of the controller and
manager tables) never used it.  Furthermore, the other callers
would be fine always setting the DSCP bits to zero.  This patch
removes the DSCP_INVALID option in an effort to simplify the code.

Signed-off-by: Ethan Jackson <ethan@nicira.com>
2012-04-17 13:15:17 -07:00

493 lines
14 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright (c) 2008, 2009, 2010, 2011, 2012 Nicira Networks.
*
* 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 <config.h>
#include "unixctl.h"
#include <assert.h>
#include <errno.h>
#include <unistd.h>
#include "coverage.h"
#include "dirs.h"
#include "dynamic-string.h"
#include "json.h"
#include "jsonrpc.h"
#include "list.h"
#include "poll-loop.h"
#include "shash.h"
#include "stream.h"
#include "svec.h"
#include "vlog.h"
VLOG_DEFINE_THIS_MODULE(unixctl);
COVERAGE_DEFINE(unixctl_received);
COVERAGE_DEFINE(unixctl_replied);
struct unixctl_command {
const char *usage;
int min_args, max_args;
unixctl_cb_func *cb;
void *aux;
};
struct unixctl_conn {
struct list node;
struct jsonrpc *rpc;
/* Only one request can be in progress at a time. While the request is
* being processed, 'request_id' is populated, otherwise it is null. */
struct json *request_id; /* ID of the currently active request. */
};
/* Server for control connection. */
struct unixctl_server {
struct pstream *listener;
struct list conns;
};
static struct vlog_rate_limit rl = VLOG_RATE_LIMIT_INIT(5, 5);
static struct shash commands = SHASH_INITIALIZER(&commands);
static void
unixctl_help(struct unixctl_conn *conn, int argc OVS_UNUSED,
const char *argv[] OVS_UNUSED, void *aux OVS_UNUSED)
{
struct ds ds = DS_EMPTY_INITIALIZER;
const struct shash_node **nodes = shash_sort(&commands);
size_t i;
ds_put_cstr(&ds, "The available commands are:\n");
for (i = 0; i < shash_count(&commands); i++) {
const struct shash_node *node = nodes[i];
const struct unixctl_command *command = node->data;
ds_put_format(&ds, " %-23s %s\n", node->name, command->usage);
}
free(nodes);
unixctl_command_reply(conn, ds_cstr(&ds));
ds_destroy(&ds);
}
static void
unixctl_version(struct unixctl_conn *conn, int argc OVS_UNUSED,
const char *argv[] OVS_UNUSED, void *aux OVS_UNUSED)
{
unixctl_command_reply(conn, get_program_version());
}
/* Registers a unixctl command with the given 'name'. 'usage' describes the
* arguments to the command; it is used only for presentation to the user in
* "help" output.
*
* 'cb' is called when the command is received. It is passed the actual set of
* arguments, as a text string, plus a copy of 'aux'. Normally 'cb' should
* reply by calling unixctl_command_reply() or unixctl_command_reply_error()
* before it returns, but if the command cannot be handled immediately then it
* can defer the reply until later. A given connection can only process a
* single request at a time, so a reply must be made eventually to avoid
* blocking that connection. */
void
unixctl_command_register(const char *name, const char *usage,
int min_args, int max_args,
unixctl_cb_func *cb, void *aux)
{
struct unixctl_command *command;
struct unixctl_command *lookup = shash_find_data(&commands, name);
assert(!lookup || lookup->cb == cb);
if (lookup) {
return;
}
command = xmalloc(sizeof *command);
command->usage = usage;
command->min_args = min_args;
command->max_args = max_args;
command->cb = cb;
command->aux = aux;
shash_add(&commands, name, command);
}
static void
unixctl_command_reply__(struct unixctl_conn *conn,
bool success, const char *body)
{
struct json *body_json;
struct jsonrpc_msg *reply;
COVERAGE_INC(unixctl_replied);
assert(conn->request_id);
if (!body) {
body = "";
}
if (body[0] && body[strlen(body) - 1] != '\n') {
body_json = json_string_create_nocopy(xasprintf("%s\n", body));
} else {
body_json = json_string_create(body);
}
if (success) {
reply = jsonrpc_create_reply(body_json, conn->request_id);
} else {
reply = jsonrpc_create_error(body_json, conn->request_id);
}
/* If jsonrpc_send() returns an error, the run loop will take care of the
* problem eventually. */
jsonrpc_send(conn->rpc, reply);
json_destroy(conn->request_id);
conn->request_id = NULL;
}
/* Replies to the active unixctl connection 'conn'. 'result' is sent to the
* client indicating the command was processed successfully. Only one call to
* unixctl_command_reply() or unixctl_command_reply_error() may be made per
* request. */
void
unixctl_command_reply(struct unixctl_conn *conn, const char *result)
{
unixctl_command_reply__(conn, true, result);
}
/* Replies to the active unixctl connection 'conn'. 'error' is sent to the
* client indicating an error occured processing the command. Only one call to
* unixctl_command_reply() or unixctl_command_reply_error() may be made per
* request. */
void
unixctl_command_reply_error(struct unixctl_conn *conn, const char *error)
{
unixctl_command_reply__(conn, false, error);
}
/* Creates a unixctl server listening on 'path', which may be:
*
* - NULL, in which case <rundir>/<program>.<pid>.ctl is used.
*
* - "none", in which case the function will return successfully but
* no socket will actually be created.
*
* - A name that does not start with '/', in which case it is put in
* <rundir>.
*
* - An absolute path (starting with '/') that gives the exact name of
* the Unix domain socket to listen on.
*
* A program that (optionally) daemonizes itself should call this function
* *after* daemonization, so that the socket name contains the pid of the
* daemon instead of the pid of the program that exited. (Otherwise,
* "ovs-appctl --target=<program>" will fail.)
*
* Returns 0 if successful, otherwise a positive errno value. If successful,
* sets '*serverp' to the new unixctl_server (or to NULL if 'path' was "none"),
* otherwise to NULL. */
int
unixctl_server_create(const char *path, struct unixctl_server **serverp)
{
struct unixctl_server *server;
struct pstream *listener;
char *punix_path;
int error;
*serverp = NULL;
if (path && !strcmp(path, "none")) {
return 0;
}
if (path) {
char *abs_path = abs_file_name(ovs_rundir(), path);
punix_path = xasprintf("punix:%s", abs_path);
free(abs_path);
} else {
punix_path = xasprintf("punix:%s/%s.%ld.ctl", ovs_rundir(),
program_name, (long int) getpid());
}
error = pstream_open(punix_path, &listener, 0);
if (error) {
ovs_error(error, "could not initialize control socket %s", punix_path);
goto exit;
}
unixctl_command_register("help", "", 0, 0, unixctl_help, NULL);
unixctl_command_register("version", "", 0, 0, unixctl_version, NULL);
server = xmalloc(sizeof *server);
server->listener = listener;
list_init(&server->conns);
*serverp = server;
exit:
free(punix_path);
return error;
}
static void
process_command(struct unixctl_conn *conn, struct jsonrpc_msg *request)
{
char *error = NULL;
struct unixctl_command *command;
struct json_array *params;
COVERAGE_INC(unixctl_received);
conn->request_id = json_clone(request->id);
params = json_array(request->params);
command = shash_find_data(&commands, request->method);
if (!command) {
error = xasprintf("\"%s\" is not a valid command", request->method);
} else if (params->n < command->min_args) {
error = xasprintf("\"%s\" command requires at least %d arguments",
request->method, command->min_args);
} else if (params->n > command->max_args) {
error = xasprintf("\"%s\" command takes at most %d arguments",
request->method, command->max_args);
} else {
struct svec argv = SVEC_EMPTY_INITIALIZER;
int i;
svec_add(&argv, request->method);
for (i = 0; i < params->n; i++) {
if (params->elems[i]->type != JSON_STRING) {
error = xasprintf("\"%s\" command has non-string argument",
request->method);
break;
}
svec_add(&argv, json_string(params->elems[i]));
}
svec_terminate(&argv);
if (!error) {
command->cb(conn, argv.n, (const char **) argv.names,
command->aux);
}
svec_destroy(&argv);
}
if (error) {
unixctl_command_reply_error(conn, error);
free(error);
}
}
static int
run_connection(struct unixctl_conn *conn)
{
int error, i;
jsonrpc_run(conn->rpc);
error = jsonrpc_get_status(conn->rpc);
if (error || jsonrpc_get_backlog(conn->rpc)) {
return error;
}
for (i = 0; i < 10; i++) {
struct jsonrpc_msg *msg;
if (error || conn->request_id) {
break;
}
jsonrpc_recv(conn->rpc, &msg);
if (msg) {
if (msg->type == JSONRPC_REQUEST) {
process_command(conn, msg);
} else {
VLOG_WARN_RL(&rl, "%s: received unexpected %s message",
jsonrpc_get_name(conn->rpc),
jsonrpc_msg_type_to_string(msg->type));
error = EINVAL;
}
jsonrpc_msg_destroy(msg);
}
error = error ? error : jsonrpc_get_status(conn->rpc);
}
return error;
}
static void
kill_connection(struct unixctl_conn *conn)
{
list_remove(&conn->node);
jsonrpc_close(conn->rpc);
json_destroy(conn->request_id);
free(conn);
}
void
unixctl_server_run(struct unixctl_server *server)
{
struct unixctl_conn *conn, *next;
int i;
if (!server) {
return;
}
for (i = 0; i < 10; i++) {
struct stream *stream;
int error;
error = pstream_accept(server->listener, &stream);
if (!error) {
struct unixctl_conn *conn = xzalloc(sizeof *conn);
list_push_back(&server->conns, &conn->node);
conn->rpc = jsonrpc_open(stream);
} else if (error == EAGAIN) {
break;
} else {
VLOG_WARN_RL(&rl, "%s: accept failed: %s",
pstream_get_name(server->listener),
strerror(error));
}
}
LIST_FOR_EACH_SAFE (conn, next, node, &server->conns) {
int error = run_connection(conn);
if (error && error != EAGAIN) {
kill_connection(conn);
}
}
}
void
unixctl_server_wait(struct unixctl_server *server)
{
struct unixctl_conn *conn;
if (!server) {
return;
}
pstream_wait(server->listener);
LIST_FOR_EACH (conn, node, &server->conns) {
jsonrpc_wait(conn->rpc);
if (!jsonrpc_get_backlog(conn->rpc)) {
jsonrpc_recv_wait(conn->rpc);
}
}
}
/* Destroys 'server' and stops listening for connections. */
void
unixctl_server_destroy(struct unixctl_server *server)
{
if (server) {
struct unixctl_conn *conn, *next;
LIST_FOR_EACH_SAFE (conn, next, node, &server->conns) {
kill_connection(conn);
}
pstream_close(server->listener);
free(server);
}
}
/* Connects to a unixctl server socket. 'path' should be the name of a unixctl
* server socket. If it does not start with '/', it will be prefixed with the
* rundir (e.g. /usr/local/var/run/openvswitch).
*
* Returns 0 if successful, otherwise a positive errno value. If successful,
* sets '*client' to the new jsonrpc, otherwise to NULL. */
int
unixctl_client_create(const char *path, struct jsonrpc **client)
{
char *abs_path, *unix_path;
struct stream *stream;
int error;
*client = NULL;
abs_path = abs_file_name(ovs_rundir(), path);
unix_path = xasprintf("unix:%s", abs_path);
error = stream_open_block(stream_open(unix_path, &stream, DSCP_DEFAULT),
&stream);
free(unix_path);
free(abs_path);
if (error) {
VLOG_WARN("failed to connect to %s", path);
return error;
}
*client = jsonrpc_open(stream);
return 0;
}
/* Executes 'command' on the server with an argument vector 'argv' containing
* 'argc' elements. If successfully communicated with the server, returns 0
* and sets '*result', or '*err' (not both) to the result or error the server
* returned. Otherwise, sets '*result' and '*err' to NULL and returns a
* positive errno value. The caller is responsible for freeing '*result' or
* '*err' if not NULL. */
int
unixctl_client_transact(struct jsonrpc *client, const char *command, int argc,
char *argv[], char **result, char **err)
{
struct jsonrpc_msg *request, *reply;
struct json **json_args, *params;
int error, i;
*result = NULL;
*err = NULL;
json_args = xmalloc(argc * sizeof *json_args);
for (i = 0; i < argc; i++) {
json_args[i] = json_string_create(argv[i]);
}
params = json_array_create(json_args, argc);
request = jsonrpc_create_request(command, params, NULL);
error = jsonrpc_transact_block(client, request, &reply);
if (error) {
VLOG_WARN("error communicating with %s: %s", jsonrpc_get_name(client),
strerror(error));
return error;
}
if (reply->error) {
if (reply->error->type == JSON_STRING) {
*err = xstrdup(json_string(reply->error));
} else {
VLOG_WARN("%s: unexpected error type in JSON RPC reply: %s",
jsonrpc_get_name(client),
json_type_to_string(reply->error->type));
error = EINVAL;
}
} else if (reply->result) {
if (reply->result->type == JSON_STRING) {
*result = xstrdup(json_string(reply->result));
} else {
VLOG_WARN("%s: unexpected result type in JSON rpc reply: %s",
jsonrpc_get_name(client),
json_type_to_string(reply->result->type));
error = EINVAL;
}
}
jsonrpc_msg_destroy(reply);
return error;
}