summaryrefslogtreecommitdiff
path: root/src/devreader-hpux.cpp
diff options
context:
space:
mode:
authorOlivier Gayot <duskcoder@gmail.com>2014-06-18 15:45:45 +0200
committerOlivier Gayot <duskcoder@gmail.com>2014-06-18 15:49:38 +0200
commit0e03940802cebefdf6b0597a154bd9395e1af4d2 (patch)
tree409a58499128227dd57943515d003074190551f5 /src/devreader-hpux.cpp
Add the vanilla version of the project
This version can still be found here: http://www.roland-riegel.de/nload/index.html
Diffstat (limited to 'src/devreader-hpux.cpp')
-rw-r--r--src/devreader-hpux.cpp294
1 files changed, 294 insertions, 0 deletions
diff --git a/src/devreader-hpux.cpp b/src/devreader-hpux.cpp
new file mode 100644
index 0000000..c1f5403
--- /dev/null
+++ b/src/devreader-hpux.cpp
@@ -0,0 +1,294 @@
+/***************************************************************************
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU General Public License as published by *
+ * the Free Software Foundation; either version 2 of the License, or *
+ * (at your option) any later version. *
+ * *
+ ***************************************************************************/
+
+/***************************************************************************
+ * HP-UX specific network code *
+ * *
+ * Copyright 1998 by Hewlett-Packard Company *
+ * *
+ * Permission to use, copy, modify, and distribute this *
+ * software and its documentation for any purpose and without *
+ * fee is hereby granted, provided that the above copyright *
+ * notice appear in all copies and that both that copyright *
+ * notice and this permission notice appear in supporting *
+ * documentation, and that the name of Hewlett-Packard Company not *
+ * be used in advertising or publicity pertaining to distribution *
+ * of the software without specific, written prior permission. *
+ * Hewlett-Packard makes no representations about the *
+ * suitability of this software for any purpose. It is provided *
+ * "as is" without express or implied warranty. *
+ * *
+ * *
+ * Modification History: *
+ * *
+ * int_stats.c : Get network statistics from the dlpi device *
+ * On 10.20 only get a mib_ifEntry, on 11.0 this is followed *
+ * by an mib_Dot3StatsEntry. Accessing the latter structure *
+ * is left as an exercise for the reader, the struct is defined *
+ * in /usr/include/sys/mib.h along with mib_ifEntry. *
+ * *
+ * Author: Jon Dewis, SPP 3/26/98 *
+ * (With ackn to 'lanadmin' source code and streams class notes) *
+ * *
+ * 24-June-1998 Add mib3 stats (AJD) when compiling *
+ * for 11.0 ensure HP_UX11 macro defined *
+ * e.g. cc -o int_stats -DHP_UX11 int_stats.c *
+ * 16-Dec-1999 Change to obviate need for lanscan to get *
+ * nmid - thus assumes any old nmid will do *
+ * and just iterates thru the 1st 100 possibilities *
+ * (MAX_NMID). Works on 10.20 and 11.0 *
+ * Dec 2003 Modified by Roshan Sequeira roshan.sequeira@hp.com, *
+ * to get network statistics for the nload port to HP-UX. *
+ * Original code available at *
+ * http://h21007.www2.hp.com/dspp/tech/tech_TechDocumentDetailPage_IDX/1,1701,2599,00.html
+ * *
+ ***************************************************************************/
+
+#include "devreader-hpux.h"
+#include "stringutils.h"
+
+#ifdef HAVE_CONFIG_H
+ #include <config.h>
+#endif
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+#include <sys/types.h>
+#include <sys/ioctl.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#include <net/if.h>
+#include <unistd.h>
+#include <fcntl.h>
+#include <sys/stropts.h>
+#include <sys/dlpi.h>
+#include <sys/dlpi_ext.h>
+#include <sys/mib.h>
+
+#include <string>
+#include <list>
+
+using namespace std;
+
+#define AREA_SIZE 40000
+#define LONG_AREA_SIZE (AREA_SIZE / sizeof(u_long))
+
+static u_long ctrl_area[LONG_AREA_SIZE]; /* for control messages */
+static u_long data_area[LONG_AREA_SIZE]; /* for data messages */
+static u_long ppa_area[LONG_AREA_SIZE]; /* for saving ppa area */
+static struct strbuf ctrl_buf = {AREA_SIZE, 0, (char*) ctrl_area};
+static struct strbuf data_buf = {AREA_SIZE, 0, (char*) data_area};
+
+DevReaderHpux::DevReaderHpux(const string& deviceName)
+ : DevReader(deviceName)
+{
+}
+
+DevReaderHpux::~DevReaderHpux()
+{
+}
+
+list<string> DevReaderHpux::findAllDevices()
+{
+ list<string> devicesFound;
+
+ dl_hp_ppa_info_t* ppa_info_temp;
+ dl_hp_ppa_req_t* ppa_req;
+ dl_hp_ppa_ack_t* ppa_ack;
+ dl_attach_req_t* attach_req;
+
+ int fd = -1;
+ if((fd = open("/dev/dlpi", O_RDWR)) < 0)
+ {
+ perror("open /dev/dlpi");
+ return devicesFound;
+ }
+
+ do
+ {
+ ppa_req = (dl_hp_ppa_req_t*) ctrl_area;
+ ppa_ack = (dl_hp_ppa_ack_t*) ctrl_area;
+
+ ppa_req->dl_primitive = DL_HP_PPA_REQ;
+
+ ctrl_buf.len = sizeof(dl_hp_ppa_req_t);
+ if(putmsg(fd, &ctrl_buf, 0, 0) < 0)
+ {
+ perror("putmsg DL_HP_PPA_REQ");
+ break;
+ }
+
+ int flags = 0;
+ ctrl_area[0] = 0;
+
+ if(getmsg(fd, &ctrl_buf, &data_buf, &flags) < 0)
+ {
+ perror("getmsg DL_HP_PPA_REQ");
+ break;
+ }
+
+ if(ppa_ack->dl_length == 0)
+ {
+ fprintf(stderr, "error: no PPAs available\n");
+ break;
+ }
+
+ // save all the PPA information
+ memcpy((u_char*) ppa_area, (u_char*) ctrl_area + ppa_ack->dl_offset, ppa_ack->dl_length);
+ int ppa_count = ppa_ack->dl_count;
+
+ int count;
+ for(count = 0, ppa_info_temp = (dl_hp_ppa_info_t*) ppa_area; count < ppa_count; count++, ppa_info_temp++)
+ devicesFound.push_back(string("lan") + toString(ppa_info_temp->dl_ppa));
+
+ } while(0);
+
+ close(fd);
+
+ return devicesFound;
+}
+
+void DevReaderHpux::readFromDevice(DataFrame& dataFrame)
+{
+ dl_get_statistics_req_t* get_statistics_req = (dl_get_statistics_req_t*) ctrl_area;
+ dl_get_statistics_ack_t* get_statistics_ack = (dl_get_statistics_ack_t*) ctrl_area;
+
+ dl_hp_ppa_info_t ppa_info;
+ dl_hp_ppa_info_t* ppa_info_temp;
+ dl_hp_ppa_req_t* ppa_req;
+ dl_hp_ppa_ack_t* ppa_ack;
+ dl_attach_req_t* attach_req;
+ mib_ifEntry* mib_ptr;
+
+ unsigned int ppa = 0;
+ int flags = 0, ppa_count = 0, count = 0;
+
+ // make sure interface name begins with lan
+ if(m_deviceName.length() < 3 || m_deviceName.substr(0, 3) != "lan")
+ return;
+
+ // get the PPA from the interface name
+ ppa = fromString<int>(m_deviceName.substr(3));
+ if(ppa == 0 && m_deviceName != "lan0")
+ return;
+
+ int fd = -1;
+ if((fd = open("/dev/dlpi", O_RDWR)) < 0)
+ {
+ perror("open /dev/dlpi");
+ return;
+ }
+
+ do
+ {
+ ppa_req = (dl_hp_ppa_req_t*) ctrl_area;
+ ppa_ack = (dl_hp_ppa_ack_t*) ctrl_area;
+
+ ppa_req->dl_primitive = DL_HP_PPA_REQ;
+
+ ctrl_buf.len = sizeof(dl_hp_ppa_req_t);
+ if(putmsg(fd, &ctrl_buf, 0, 0) < 0)
+ {
+ perror("putmsg DL_HP_PPA_REQ");
+ break;
+ }
+
+ flags = 0;
+ ctrl_area[0] = 0;
+
+ if(getmsg(fd, &ctrl_buf, &data_buf, &flags) < 0)
+ {
+ perror("getmsg DL_HP_PPA_REQ");
+ break;
+ }
+
+ if(ppa_ack->dl_length == 0)
+ {
+ fprintf(stderr, "Error: No PPAs available\n");
+ break;
+ }
+
+ // save all the PPA information
+ memcpy((u_char*) ppa_area, (u_char*) ctrl_area + ppa_ack->dl_offset, ppa_ack->dl_length);
+ ppa_count = ppa_ack->dl_count;
+
+ bool found = false;
+ for(count = 0, ppa_info_temp = (dl_hp_ppa_info_t*) ppa_area; count < ppa_count; count++, ppa_info_temp++)
+ {
+ if(ppa_info_temp->dl_ppa == ppa)
+ {
+ found = true;
+ break;
+ }
+ }
+ if(!found)
+ {
+ fprintf(stderr, "error: PPA %d not found\n", ppa);
+ break;
+ }
+
+ attach_req = (dl_attach_req_t*) ctrl_area;
+ attach_req->dl_primitive = DL_ATTACH_REQ;
+ attach_req->dl_ppa = ppa;
+ ctrl_buf.len = sizeof(dl_attach_req_t);
+ if(putmsg(fd, &ctrl_buf, 0, 0) < 0)
+ {
+ perror("putmsg DL_ATTACH_REQ");
+ break;
+ }
+
+ ctrl_area[0] = 0;
+ if(getmsg(fd, &ctrl_buf, &data_buf, &flags) < 0)
+ {
+ perror("getmsg DL_ATTACH_REQ");
+ break;
+ }
+ memcpy(&ppa_info, ppa_info_temp, sizeof(ppa_info));
+
+ get_statistics_req->dl_primitive = DL_GET_STATISTICS_REQ;
+ ctrl_buf.len = sizeof(dl_get_statistics_req_t);
+ flags = 0;
+
+ if(putmsg(fd, &ctrl_buf, NULL, 0) < 0)
+ {
+ perror("putmsg DL_GET_STATISTICS_REQ");
+ break;
+ }
+
+ if(getmsg(fd, &ctrl_buf, NULL, &flags) < 0)
+ {
+ perror("getmsg DL_GET_STATISTICS_REQ");
+ break;
+ }
+ if(get_statistics_ack->dl_primitive != DL_GET_STATISTICS_ACK)
+ fprintf(stderr, "error: wrong primitive\n");
+
+ mib_ptr = (mib_ifEntry*) ((u_char*) ctrl_area + get_statistics_ack->dl_stat_offset);
+
+ dataFrame.setTotalDataIn(mib_ptr->ifInOctets);
+ dataFrame.setTotalDataOut(mib_ptr->ifOutOctets);
+
+ dataFrame.setTotalPacketsIn(mib_ptr->ifInUcastPkts + mib_ptr->ifInNUcastPkts);
+ dataFrame.setTotalPacketsOut(mib_ptr->ifOutUcastPkts + mib_ptr->ifOutNUcastPkts);
+
+ dataFrame.setTotalErrorsIn(mib_ptr->ifInErrors);
+ dataFrame.setTotalErrorsOut(mib_ptr->ifOutErrors);
+
+ dataFrame.setTotalDropsIn(mib_ptr->ifInDiscards);
+ dataFrame.setTotalDropsOut(mib_ptr->ifOutDiscards);
+
+ dataFrame.setValid(true);
+
+ } while(0);
+
+ close(fd);
+}
+