/* Example program using the lowlevel raw interface.
* This allow accurate control of the exact commands that are being used.
*/
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+#ifdef WIN32
+#include "win32_compat.h"
+#endif
#define SERVER "10.1.1.27"
#define EXPORT "/shared"
+#ifdef HAVE_POLL_H
+#include <poll.h>
+#endif
+
+#ifdef HAVE_NETINET_IN_H
+#include <netinet/in.h>
+#endif
+
#include <stdio.h>
#include <stdlib.h>
-#include <poll.h>
+#include <string.h>
+#include "libnfs-zdr.h"
#include "libnfs.h"
#include "libnfs-raw.h"
#include "libnfs-raw-mount.h"
#include "libnfs-raw-nfs.h"
+#include "libnfs-raw-rquota.h"
struct client {
char *server;
char *export;
uint32_t mount_port;
+ uint32_t rquota_port;
int is_finished;
struct nfs_fh3 rootfh;
};
+void rquota_getquota_cb(struct rpc_context *rpc _U_, int status, void *data, void *private_data)
+{
+ struct client *client = private_data;
+// GETQUOTA1res *res = data;
+
+ if (status == RPC_STATUS_ERROR) {
+ printf("rquota/getquota call failed with \"%s\"\n", (char *)data);
+ exit(10);
+ }
+ if (status != RPC_STATUS_SUCCESS) {
+ printf("rquota/getquota call to server %s failed, status:%d\n", client->server, status);
+ exit(10);
+ }
+
+ printf("rquota responded ok\n");
+ client->is_finished = 1;
+}
+
+void rquota_connect_cb(struct rpc_context *rpc, int status, void *data _U_, void *private_data)
+{
+ struct client *client = private_data;
+
+ if (status != RPC_STATUS_SUCCESS) {
+ printf("connection to RPC.RQUOTAD on server %s failed\n", client->server);
+ exit(10);
+ }
+
+ printf("Connected to RPC.RQUOTAD on %s:%d\n", client->server, client->rquota_port);
+ printf("Send GETQUOTA request for uid 100\n");
+ if (rpc_rquota1_getquota_async(rpc, rquota_getquota_cb, EXPORT, 100, client) != 0) {
+ printf("Failed to send fsinfo request\n");
+ exit(10);
+ }
+}
+
+void acl_getacl_cb(struct rpc_context *rpc _U_, int status, void *data, void *private_data)
+{
+ struct client *client = private_data;
+ GETACL3res *res = data;
+
+ printf("Got NFSACL/GETACL reply\n");
+
+ if (status == RPC_STATUS_SUCCESS) {
+ printf("Got an ACL : ACL status:%d\n", res->status);
+ if (res->status == NFS3_OK) {
+ int i;
+ printf("ACL MASK 0x%08x\n", res->GETACL3res_u.resok.mask);
+ printf("NUM ACE %d\n", res->GETACL3res_u.resok.ace_count);
+ for (i=0; i<res->GETACL3res_u.resok.ace.ace_len; i++) {
+ printf("Type:0x%08x\n", res->GETACL3res_u.resok.ace.ace_val[i].type);
+ printf("ID:%d\n", res->GETACL3res_u.resok.ace.ace_val[i].id);
+ printf("Perm:0x%08x\n", res->GETACL3res_u.resok.ace.ace_val[i].perm);
+ }
+ }
+ }
+
+ printf("Disconnect socket from nfs server\n");
+ if (rpc_disconnect(rpc, "normal disconnect") != 0) {
+ printf("Failed to disconnect socket to nfs\n");
+ exit(10);
+ }
+
+ printf("Connect to RPC.RQUOTAD on %s:%d\n", client->server, client->rquota_port);
+ if (rpc_connect_async(rpc, client->server, client->rquota_port, rquota_connect_cb, client) != 0) {
+ printf("Failed to start connection\n");
+ exit(10);
+ }
+}
+
+void acl_null_cb(struct rpc_context *rpc _U_, int status, void *data, void *private_data)
+{
+ struct client *client = private_data;
+ GETACL3args args;
+
+ printf("Got NFSACL/NULL reply\n");
+ printf("Get ACL for root handle\n");
+
+ args.dir = client->rootfh;
+ args.mask = NFSACL_MASK_ACL_ENTRY|NFSACL_MASK_ACL_COUNT|NFSACL_MASK_ACL_DEFAULT_ENTRY|NFSACL_MASK_ACL_DEFAULT_COUNT;
+ if (rpc_nfsacl_getacl_async(rpc, acl_getacl_cb, &args, client) != 0) {
+ printf("Failed to send getacl request\n");
+ exit(10);
+ }
+
+}
+
void nfs_fsinfo_cb(struct rpc_context *rpc _U_, int status, void *data, void *private_data)
{
struct client *client = private_data;
printf("Got reply from server for NFS/FSINFO procedure.\n");
printf("Read Max:%d\n", (int)res->FSINFO3res_u.resok.rtmax);
printf("Write Max:%d\n", (int)res->FSINFO3res_u.resok.wtmax);
- client->is_finished = 1;
+
+ printf("Send NFSACL/NULL request\n");
+ if (rpc_nfsacl_null_async(rpc, acl_null_cb, client) != 0) {
+ printf("Failed to send acl/null request\n");
+ exit(10);
+ }
}
exit(10);
}
- printf("Connected to RPC.NFSDD on %s:%d\n", client->server, client->mount_port);
+ printf("Connected to RPC.NFSD on %s:%d\n", client->server, client->mount_port);
printf("Send FSINFO request\n");
if (rpc_nfs_fsinfo_async(rpc, nfs_fsinfo_cb, &client->rootfh, client) != 0) {
printf("Failed to send fsinfo request\n");
exit(10);
}
- printf("Connect to RPC.NFSDD on %s:%d\n", client->server, 2049);
+ printf("Connect to RPC.NFSD on %s:%d\n", client->server, 2049);
if (rpc_connect_async(rpc, client->server, 2049, nfs_connect_cb, client) != 0) {
printf("Failed to start connection\n");
exit(10);
}
-void mount_null_cb(struct rpc_context *rpc, int status, void *data, void *private_data)
+
+void mount_export_cb(struct rpc_context *rpc, int status, void *data, void *private_data)
{
struct client *client = private_data;
+ exports export = *(exports *)data;
if (status == RPC_STATUS_ERROR) {
printf("mount null call failed with \"%s\"\n", (char *)data);
exit(10);
}
- printf("Got reply from server for MOUNT/NULL procedure.\n");
+ printf("Got reply from server for MOUNT/EXPORT procedure.\n");
+ while (export != NULL) {
+ printf("Export: %s\n", export->ex_dir);
+ export = export->ex_next;
+ }
printf("Send MOUNT/MNT command for %s\n", client->export);
if (rpc_mount_mnt_async(rpc, mount_mnt_cb, client->export, client) != 0) {
printf("Failed to send mnt request\n");
}
}
+void mount_null_cb(struct rpc_context *rpc, int status, void *data, void *private_data)
+{
+ struct client *client = private_data;
+
+ if (status == RPC_STATUS_ERROR) {
+ printf("mount null call failed with \"%s\"\n", (char *)data);
+ exit(10);
+ }
+ if (status != RPC_STATUS_SUCCESS) {
+ printf("mount null call to server %s failed, status:%d\n", client->server, status);
+ exit(10);
+ }
+
+ printf("Got reply from server for MOUNT/NULL procedure.\n");
+ printf("Send MOUNT/EXPORT command\n");
+ if (rpc_mount_export_async(rpc, mount_export_cb, client) != 0) {
+ printf("Failed to send export request\n");
+ exit(10);
+ }
+}
+
void mount_connect_cb(struct rpc_context *rpc, int status, void *data _U_, void *private_data)
{
struct client *client = private_data;
}
-void pmap_getport_cb(struct rpc_context *rpc, int status, void *data, void *private_data)
+void pmap_getport2_cb(struct rpc_context *rpc, int status, void *data, void *private_data)
{
struct client *client = private_data;
}
client->mount_port = *(uint32_t *)data;
- printf("GETPORT returned Port:%d\n", client->mount_port);
+ printf("GETPORT returned RPC.MOUNTD is on port:%d\n", client->mount_port);
if (client->mount_port == 0) {
printf("RPC.MOUNTD is not available on server : %s:%d\n", client->server, client->mount_port);
exit(10);
}
}
+void pmap_getport1_cb(struct rpc_context *rpc, int status, void *data, void *private_data)
+{
+ struct client *client = private_data;
+
+ if (status == RPC_STATUS_ERROR) {
+ printf("portmapper getport call failed with \"%s\"\n", (char *)data);
+ exit(10);
+ }
+ if (status != RPC_STATUS_SUCCESS) {
+ printf("portmapper getport call to server %s failed, status:%d\n", client->server, status);
+ exit(10);
+ }
+
+ client->rquota_port = *(uint32_t *)data;
+ printf("GETPORT returned RPC.RQUOTAD on port:%d\n", client->rquota_port);
+ if (client->rquota_port == 0) {
+ printf("RPC.RQUOTAD is not available on server : %s:%d\n", client->server, client->rquota_port);
+// exit(10);
+ }
+
+ printf("Send getport request asking for MOUNT port\n");
+ if (rpc_pmap_getport_async(rpc, MOUNT_PROGRAM, MOUNT_V3, IPPROTO_TCP, pmap_getport2_cb, client) != 0) {
+ printf("Failed to send getport request\n");
+ exit(10);
+ }
+}
void pmap_null_cb(struct rpc_context *rpc, int status, void *data, void *private_data)
{
printf("Got reply from server for PORTMAP/NULL procedure.\n");
printf("Send getport request asking for MOUNT port\n");
- if (rpc_pmap_getport_async(rpc, MOUNT_PROGRAM, MOUNT_V3, pmap_getport_cb, client) != 0) {
+ if (rpc_pmap_getport_async(rpc, RQUOTA_PROGRAM, RQUOTA_V1, IPPROTO_TCP, pmap_getport1_cb, client) != 0) {
printf("Failed to send getport request\n");
exit(10);
}