-rw-r--r-- | libopie2/opienet/opcap.cpp | 42 | ||||
-rw-r--r-- | libopie2/opienet/opcap.h | 21 |
2 files changed, 48 insertions, 15 deletions
diff --git a/libopie2/opienet/opcap.cpp b/libopie2/opienet/opcap.cpp index bef9182..1de7124 100644 --- a/libopie2/opienet/opcap.cpp +++ b/libopie2/opienet/opcap.cpp | |||
@@ -819,208 +819,228 @@ OWaveLanControlPacket::~OWaveLanControlPacket() | |||
819 | 819 | ||
820 | 820 | ||
821 | /*====================================================================================== | 821 | /*====================================================================================== |
822 | * OPacketCapturer | 822 | * OPacketCapturer |
823 | *======================================================================================*/ | 823 | *======================================================================================*/ |
824 | 824 | ||
825 | OPacketCapturer::OPacketCapturer( QObject* parent, const char* name ) | 825 | OPacketCapturer::OPacketCapturer( QObject* parent, const char* name ) |
826 | :QObject( parent, name ), _name( QString::null ), _open( false ), | 826 | :QObject( parent, name ), _name( QString::null ), _open( false ), |
827 | _pch( 0 ), _pcd( 0 ), _sn( 0 ) | 827 | _pch( 0 ), _pcd( 0 ), _sn( 0 ) |
828 | { | 828 | { |
829 | } | 829 | } |
830 | 830 | ||
831 | 831 | ||
832 | OPacketCapturer::~OPacketCapturer() | 832 | OPacketCapturer::~OPacketCapturer() |
833 | { | 833 | { |
834 | if ( _open ) | 834 | if ( _open ) |
835 | { | 835 | { |
836 | qDebug( "OPacketCapturer::~OPacketCapturer(): pcap still open, autoclosing." ); | 836 | qDebug( "OPacketCapturer::~OPacketCapturer(): pcap still open, autoclosing." ); |
837 | close(); | 837 | close(); |
838 | } | 838 | } |
839 | } | 839 | } |
840 | 840 | ||
841 | 841 | ||
842 | void OPacketCapturer::setBlocking( bool b ) | 842 | void OPacketCapturer::setBlocking( bool b ) |
843 | { | 843 | { |
844 | if ( pcap_setnonblock( _pch, 1-b, _errbuf ) != -1 ) | 844 | if ( pcap_setnonblock( _pch, 1-b, _errbuf ) != -1 ) |
845 | { | 845 | { |
846 | qDebug( "OPacketCapturer::setBlocking(): blocking mode changed successfully." ); | 846 | qDebug( "OPacketCapturer::setBlocking(): blocking mode changed successfully." ); |
847 | } | 847 | } |
848 | else | 848 | else |
849 | { | 849 | { |
850 | qDebug( "OPacketCapturer::setBlocking(): can't change blocking mode: %s", _errbuf ); | 850 | qDebug( "OPacketCapturer::setBlocking(): can't change blocking mode: %s", _errbuf ); |
851 | } | 851 | } |
852 | } | 852 | } |
853 | 853 | ||
854 | 854 | ||
855 | bool OPacketCapturer::blocking() const | 855 | bool OPacketCapturer::blocking() const |
856 | { | 856 | { |
857 | int b = pcap_getnonblock( _pch, _errbuf ); | 857 | int b = pcap_getnonblock( _pch, _errbuf ); |
858 | if ( b == -1 ) | 858 | if ( b == -1 ) |
859 | { | 859 | { |
860 | qDebug( "OPacketCapturer::blocking(): can't get blocking mode: %s", _errbuf ); | 860 | qDebug( "OPacketCapturer::blocking(): can't get blocking mode: %s", _errbuf ); |
861 | return -1; | 861 | return -1; |
862 | } | 862 | } |
863 | return !b; | 863 | return !b; |
864 | } | 864 | } |
865 | 865 | ||
866 | 866 | ||
867 | void OPacketCapturer::closeDumpFile() | ||
868 | { | ||
869 | if ( _pcd ) | ||
870 | { | ||
871 | pcap_dump_close( _pcd ); | ||
872 | _pcd = 0; | ||
873 | } | ||
874 | pcap_close( _pch ); | ||
875 | } | ||
876 | |||
877 | |||
867 | void OPacketCapturer::close() | 878 | void OPacketCapturer::close() |
868 | { | 879 | { |
869 | if ( _open ) | 880 | if ( _open ) |
870 | { | 881 | { |
871 | if ( _sn ) | 882 | if ( _sn ) |
872 | { | 883 | { |
873 | _sn->disconnect( SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); | 884 | _sn->disconnect( SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); |
874 | delete _sn; | 885 | delete _sn; |
875 | } | 886 | } |
876 | if ( _pcd ) | 887 | closeDumpFile(); |
877 | { | 888 | _open = false; |
878 | pcap_dump_close( _pcd ); | ||
879 | _pcd = 0; | ||
880 | } | ||
881 | pcap_close( _pch ); | ||
882 | _open = false; | ||
883 | } | 889 | } |
884 | 890 | ||
885 | qDebug( "OPacketCapturer::close() --- dumping capturing statistics..." ); | 891 | qDebug( "OPacketCapturer::close() --- dumping capturing statistics..." ); |
886 | qDebug( "--------------------------------------------------" ); | 892 | qDebug( "--------------------------------------------------" ); |
887 | for( QMap<QString,int>::Iterator it = _stats.begin(); it != _stats.end(); ++it ) | 893 | for( QMap<QString,int>::Iterator it = _stats.begin(); it != _stats.end(); ++it ) |
888 | qDebug( "%s : %d", (const char*) it.key(), it.data() ); | 894 | qDebug( "%s : %d", (const char*) it.key(), it.data() ); |
889 | qDebug( "--------------------------------------------------" ); | 895 | qDebug( "--------------------------------------------------" ); |
890 | 896 | ||
891 | } | 897 | } |
892 | 898 | ||
893 | 899 | ||
894 | int OPacketCapturer::dataLink() const | 900 | int OPacketCapturer::dataLink() const |
895 | { | 901 | { |
896 | return pcap_datalink( _pch ); | 902 | return pcap_datalink( _pch ); |
897 | } | 903 | } |
898 | 904 | ||
899 | 905 | ||
906 | void OPacketCapturer::dump( OPacket* p ) | ||
907 | { | ||
908 | if ( !_pcd ) | ||
909 | { | ||
910 | qWarning( "OPacketCapturer::dump() - cannot dump without open capture file!" ); | ||
911 | return; | ||
912 | } | ||
913 | pcap_dump( (u_char*) _pcd, &p->_hdr, p->_data ); | ||
914 | } | ||
915 | |||
916 | |||
900 | int OPacketCapturer::fileno() const | 917 | int OPacketCapturer::fileno() const |
901 | { | 918 | { |
902 | if ( _open ) | 919 | if ( _open ) |
903 | { | 920 | { |
904 | return pcap_fileno( _pch ); | 921 | return pcap_fileno( _pch ); |
905 | } | 922 | } |
906 | else | 923 | else |
907 | { | 924 | { |
908 | return -1; | 925 | return -1; |
909 | } | 926 | } |
910 | } | 927 | } |
911 | 928 | ||
912 | OPacket* OPacketCapturer::next() | 929 | OPacket* OPacketCapturer::next() |
913 | { | 930 | { |
914 | packetheaderstruct header; | 931 | packetheaderstruct header; |
915 | qDebug( "==> OPacketCapturer::next()" ); | 932 | qDebug( "==> OPacketCapturer::next()" ); |
916 | const unsigned char* pdata = pcap_next( _pch, &header ); | 933 | const unsigned char* pdata = pcap_next( _pch, &header ); |
917 | qDebug( "<== OPacketCapturer::next()" ); | 934 | qDebug( "<== OPacketCapturer::next()" ); |
918 | if ( _pcd ) | ||
919 | pcap_dump( (u_char*) _pcd, &header, pdata ); | ||
920 | 935 | ||
921 | if ( pdata && header.len ) | 936 | if ( pdata && header.len ) |
922 | { | 937 | { |
923 | OPacket* p = new OPacket( dataLink(), header, pdata, 0 ); | 938 | OPacket* p = new OPacket( dataLink(), header, pdata, 0 ); |
924 | // packets shouldn't be inserted in the QObject child-parent hierarchy, | 939 | // packets shouldn't be inserted in the QObject child-parent hierarchy, |
925 | // because due to memory constraints they will be deleted as soon | 940 | // because due to memory constraints they will be deleted as soon |
926 | // as possible - that is right after they have been processed | 941 | // as possible - that is right after they have been processed |
927 | // by emit() [ see below ] | 942 | // by emit() [ see below ] |
928 | //TODO: make gathering statistics optional, because it takes time | 943 | //TODO: make gathering statistics optional, because it takes time |
929 | p->updateStats( _stats, const_cast<QObjectList*>( p->children() ) ); | 944 | p->updateStats( _stats, const_cast<QObjectList*>( p->children() ) ); |
930 | 945 | ||
931 | return p; | 946 | return p; |
932 | } | 947 | } |
933 | else | 948 | else |
934 | { | 949 | { |
935 | qWarning( "OPacketCapturer::next() - no packet received!" ); | 950 | qWarning( "OPacketCapturer::next() - no packet received!" ); |
936 | return 0; | 951 | return 0; |
937 | } | 952 | } |
938 | } | 953 | } |
939 | 954 | ||
940 | 955 | ||
941 | bool OPacketCapturer::open( const QString& name, const QString& filename ) | 956 | bool OPacketCapturer::open( const QString& name ) |
942 | { | 957 | { |
943 | if ( _open ) | 958 | if ( _open ) |
944 | { | 959 | { |
945 | if ( name == _name ) // ignore opening an already openend device | 960 | if ( name == _name ) // ignore opening an already openend device |
946 | { | 961 | { |
947 | return true; | 962 | return true; |
948 | } | 963 | } |
949 | else // close the last opened device | 964 | else // close the last opened device |
950 | { | 965 | { |
951 | close(); | 966 | close(); |
952 | } | 967 | } |
953 | } | 968 | } |
954 | 969 | ||
955 | _name = name; | 970 | _name = name; |
956 | 971 | ||
957 | // open libpcap | 972 | // open libpcap |
958 | pcap_t* handle = pcap_open_live( const_cast<char*>( (const char*) name ), 1024, 0, 0, &_errbuf[0] ); | 973 | pcap_t* handle = pcap_open_live( const_cast<char*>( (const char*) name ), 1024, 0, 0, &_errbuf[0] ); |
959 | 974 | ||
960 | if ( !handle ) | 975 | if ( !handle ) |
961 | { | 976 | { |
962 | qWarning( "OPacketCapturer::open(): can't open libpcap with '%s': %s", (const char*) name, _errbuf ); | 977 | qWarning( "OPacketCapturer::open(): can't open libpcap with '%s': %s", (const char*) name, _errbuf ); |
963 | return false; | 978 | return false; |
964 | } | 979 | } |
965 | 980 | ||
966 | qDebug( "OPacketCapturer::open(): libpcap [%s] opened successfully.", (const char*) name ); | 981 | qDebug( "OPacketCapturer::open(): libpcap [%s] opened successfully.", (const char*) name ); |
967 | _pch = handle; | 982 | _pch = handle; |
968 | _open = true; | 983 | _open = true; |
969 | _stats.clear(); | 984 | _stats.clear(); |
970 | 985 | ||
971 | // in case we have an application object, create a socket notifier | 986 | // in case we have an application object, create a socket notifier |
972 | if ( qApp ) //TODO: I don't like this here... | 987 | if ( qApp ) //TODO: I don't like this here... |
973 | { | 988 | { |
974 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); | 989 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); |
975 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); | 990 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); |
976 | } | 991 | } |
977 | 992 | ||
978 | // if requested, open a dump | 993 | return true; |
994 | } | ||
995 | |||
996 | |||
997 | bool OPacketCapturer::openDumpFile( const QString& filename ) | ||
998 | { | ||
979 | pcap_dumper_t* dump = pcap_dump_open( _pch, const_cast<char*>( (const char*) filename ) ); | 999 | pcap_dumper_t* dump = pcap_dump_open( _pch, const_cast<char*>( (const char*) filename ) ); |
980 | if ( !dump ) | 1000 | if ( !dump ) |
981 | { | 1001 | { |
982 | qWarning( "OPacketCapturer::open(): can't open dump with '%s': %s", (const char*) filename, _errbuf ); | 1002 | qWarning( "OPacketCapturer::open(): can't open dump with '%s': %s", (const char*) filename, _errbuf ); |
983 | return false; | 1003 | return false; |
984 | } | 1004 | } |
985 | qDebug( "OPacketCapturer::open(): dump [%s] opened successfully.", (const char*) filename ); | 1005 | qDebug( "OPacketCapturer::open(): dump [%s] opened successfully.", (const char*) filename ); |
986 | _pcd = dump; | 1006 | _pcd = dump; |
987 | 1007 | ||
988 | return true; | 1008 | return true; |
989 | } | 1009 | } |
990 | 1010 | ||
991 | 1011 | ||
992 | bool OPacketCapturer::open( const QFile& file ) | 1012 | bool OPacketCapturer::open( const QFile& file ) |
993 | { | 1013 | { |
994 | QString name = file.name(); | 1014 | QString name = file.name(); |
995 | 1015 | ||
996 | if ( _open ) | 1016 | if ( _open ) |
997 | { | 1017 | { |
998 | close(); | 1018 | close(); |
999 | if ( name == _name ) // ignore opening an already openend device | 1019 | if ( name == _name ) // ignore opening an already openend device |
1000 | { | 1020 | { |
1001 | return true; | 1021 | return true; |
1002 | } | 1022 | } |
1003 | else // close the last opened device | 1023 | else // close the last opened device |
1004 | { | 1024 | { |
1005 | close(); | 1025 | close(); |
1006 | } | 1026 | } |
1007 | } | 1027 | } |
1008 | 1028 | ||
1009 | _name = name; | 1029 | _name = name; |
1010 | 1030 | ||
1011 | pcap_t* handle = pcap_open_offline( const_cast<char*>( (const char*) name ), &_errbuf[0] ); | 1031 | pcap_t* handle = pcap_open_offline( const_cast<char*>( (const char*) name ), &_errbuf[0] ); |
1012 | 1032 | ||
1013 | if ( handle ) | 1033 | if ( handle ) |
1014 | { | 1034 | { |
1015 | qDebug( "OPacketCapturer::open(): libpcap opened successfully." ); | 1035 | qDebug( "OPacketCapturer::open(): libpcap opened successfully." ); |
1016 | _pch = handle; | 1036 | _pch = handle; |
1017 | _open = true; | 1037 | _open = true; |
1018 | 1038 | ||
1019 | // in case we have an application object, create a socket notifier | 1039 | // in case we have an application object, create a socket notifier |
1020 | if ( qApp ) | 1040 | if ( qApp ) |
1021 | { | 1041 | { |
1022 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); | 1042 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); |
1023 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); | 1043 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); |
1024 | } | 1044 | } |
1025 | 1045 | ||
1026 | return true; | 1046 | return true; |
diff --git a/libopie2/opienet/opcap.h b/libopie2/opienet/opcap.h index ad5b07c..9119972 100644 --- a/libopie2/opienet/opcap.h +++ b/libopie2/opienet/opcap.h | |||
@@ -70,96 +70,98 @@ class QSocketNotifier; | |||
70 | *======================================================================================*/ | 70 | *======================================================================================*/ |
71 | 71 | ||
72 | /** @brief A class representing a data frame on the wire. | 72 | /** @brief A class representing a data frame on the wire. |
73 | * | 73 | * |
74 | * The whole family of the packet classes are used when capturing frames from a network. | 74 | * The whole family of the packet classes are used when capturing frames from a network. |
75 | * Most standard network protocols in use share a common architecture, which mostly is | 75 | * Most standard network protocols in use share a common architecture, which mostly is |
76 | * a packet header and then the packet payload. In layered architectures, each lower layer | 76 | * a packet header and then the packet payload. In layered architectures, each lower layer |
77 | * encapsulates data from its upper layer - that is it | 77 | * encapsulates data from its upper layer - that is it |
78 | * treats the data from its upper layer as payload and prepends an own header to the packet, | 78 | * treats the data from its upper layer as payload and prepends an own header to the packet, |
79 | * which - again - is treated as the payload for the layer below. The figure below is an | 79 | * which - again - is treated as the payload for the layer below. The figure below is an |
80 | * example for how such a data frame is composed out of packets, e.g. when sending a mail. | 80 | * example for how such a data frame is composed out of packets, e.g. when sending a mail. |
81 | * | 81 | * |
82 | * <pre> | 82 | * <pre> |
83 | * | User Data | == Mail Data | 83 | * | User Data | == Mail Data |
84 | * | SMTP Header | User Data | == SMTP | 84 | * | SMTP Header | User Data | == SMTP |
85 | * | TCP Header | SMTP Header | User Data | == TCP | 85 | * | TCP Header | SMTP Header | User Data | == TCP |
86 | * | IP Header | TCP Header | SMTP Header | User Data | == IP | 86 | * | IP Header | TCP Header | SMTP Header | User Data | == IP |
87 | * | MAC Header | IP Header | TCP Header | SMTP Header | User Data | == MAC | 87 | * | MAC Header | IP Header | TCP Header | SMTP Header | User Data | == MAC |
88 | * | 88 | * |
89 | * </pre> | 89 | * </pre> |
90 | * | 90 | * |
91 | * The example is trimmed for simplicity, because the MAC (Medium Access Control) layer | 91 | * The example is trimmed for simplicity, because the MAC (Medium Access Control) layer |
92 | * also contains a few more levels of encapsulation. | 92 | * also contains a few more levels of encapsulation. |
93 | * Since the type of the payload is more or less independent from the encapsulating protocol, | 93 | * Since the type of the payload is more or less independent from the encapsulating protocol, |
94 | * the header must be inspected before attempting to decode the payload. Hence, the | 94 | * the header must be inspected before attempting to decode the payload. Hence, the |
95 | * encapsulation level varies and can't be deduced without actually looking into the packets. | 95 | * encapsulation level varies and can't be deduced without actually looking into the packets. |
96 | * | 96 | * |
97 | * For actually working with captured frames, it's useful to identify the packets via names and | 97 | * For actually working with captured frames, it's useful to identify the packets via names and |
98 | * insert them into a parent/child - relationship based on the encapsulation. This is why | 98 | * insert them into a parent/child - relationship based on the encapsulation. This is why |
99 | * all packet classes derive from QObject. The amount of overhead caused by the QObject is | 99 | * all packet classes derive from QObject. The amount of overhead caused by the QObject is |
100 | * not a problem in this case, because we're talking about a theoratical maximum of about | 100 | * not a problem in this case, because we're talking about a theoratical maximum of about |
101 | * 10 packets per captured frame. We need to stuff them into a searchable list anyway and the | 101 | * 10 packets per captured frame. We need to stuff them into a searchable list anyway and the |
102 | * QObject also cares about destroying the sub-, (child-) packets. | 102 | * QObject also cares about destroying the sub-, (child-) packets. |
103 | * | 103 | * |
104 | * This enables us to perform a simple look for packets of a certain type: | 104 | * This enables us to perform a simple look for packets of a certain type: |
105 | * @code | 105 | * @code |
106 | * OPacketCapturer* pcap = new OPacketCapturer(); | 106 | * OPacketCapturer* pcap = new OPacketCapturer(); |
107 | * pcap->open( "eth0" ); | 107 | * pcap->open( "eth0" ); |
108 | * OPacket* p = pcap->next(); | 108 | * OPacket* p = pcap->next(); |
109 | * OIPPacket* ip = (OIPPacket*) p->child( "IP" ); // returns 0, if no such child exists | 109 | * OIPPacket* ip = (OIPPacket*) p->child( "IP" ); // returns 0, if no such child exists |
110 | * odebug << "got ip packet from " << ip->fromIPAddress().toString() << " to " << ip->toIPAddress().toString() << oendl; | 110 | * odebug << "got ip packet from " << ip->fromIPAddress().toString() << " to " << ip->toIPAddress().toString() << oendl; |
111 | * | 111 | * |
112 | */ | 112 | */ |
113 | 113 | ||
114 | class OPacket : public QObject | 114 | class OPacket : public QObject |
115 | { | 115 | { |
116 | Q_OBJECT | 116 | Q_OBJECT |
117 | 117 | ||
118 | friend class OPacketCapturer; | ||
119 | |||
118 | public: | 120 | public: |
119 | OPacket( int datalink, packetheaderstruct, const unsigned char*, QObject* parent ); | 121 | OPacket( int datalink, packetheaderstruct, const unsigned char*, QObject* parent ); |
120 | virtual ~OPacket(); | 122 | virtual ~OPacket(); |
121 | 123 | ||
122 | timevalstruct timeval() const; | 124 | timevalstruct timeval() const; |
123 | 125 | ||
124 | int caplen() const; | 126 | int caplen() const; |
125 | int len() const; | 127 | int len() const; |
126 | QString dump( int = 32 ) const; | 128 | QString dump( int = 32 ) const; |
127 | 129 | ||
128 | void updateStats( QMap<QString,int>&, QObjectList* ); | 130 | void updateStats( QMap<QString,int>&, QObjectList* ); |
129 | 131 | ||
130 | private: | 132 | private: |
131 | const packetheaderstruct _hdr; // pcap packet header | 133 | const packetheaderstruct _hdr; // pcap packet header |
132 | const unsigned char* _data; // pcap packet data | 134 | const unsigned char* _data; // pcap packet data |
133 | const unsigned char* _end; // end of pcap packet data | 135 | const unsigned char* _end; // end of pcap packet data |
134 | }; | 136 | }; |
135 | 137 | ||
136 | /*====================================================================================== | 138 | /*====================================================================================== |
137 | * OEthernetPacket - DLT_EN10MB frame | 139 | * OEthernetPacket - DLT_EN10MB frame |
138 | *======================================================================================*/ | 140 | *======================================================================================*/ |
139 | 141 | ||
140 | class OEthernetPacket : public QObject | 142 | class OEthernetPacket : public QObject |
141 | { | 143 | { |
142 | Q_OBJECT | 144 | Q_OBJECT |
143 | 145 | ||
144 | public: | 146 | public: |
145 | OEthernetPacket( const unsigned char*, const struct ether_header*, QObject* parent = 0 ); | 147 | OEthernetPacket( const unsigned char*, const struct ether_header*, QObject* parent = 0 ); |
146 | virtual ~OEthernetPacket(); | 148 | virtual ~OEthernetPacket(); |
147 | 149 | ||
148 | OMacAddress sourceAddress() const; | 150 | OMacAddress sourceAddress() const; |
149 | OMacAddress destinationAddress() const; | 151 | OMacAddress destinationAddress() const; |
150 | int type() const; | 152 | int type() const; |
151 | 153 | ||
152 | private: | 154 | private: |
153 | const struct ether_header* _ether; | 155 | const struct ether_header* _ether; |
154 | }; | 156 | }; |
155 | 157 | ||
156 | /*====================================================================================== | 158 | /*====================================================================================== |
157 | * OPrismHeaderPacket - DLT_PRISM_HEADER frame | 159 | * OPrismHeaderPacket - DLT_PRISM_HEADER frame |
158 | *======================================================================================*/ | 160 | *======================================================================================*/ |
159 | 161 | ||
160 | class OPrismHeaderPacket : public QObject | 162 | class OPrismHeaderPacket : public QObject |
161 | { | 163 | { |
162 | Q_OBJECT | 164 | Q_OBJECT |
163 | 165 | ||
164 | public: | 166 | public: |
165 | OPrismHeaderPacket( const unsigned char*, const struct prism_hdr*, QObject* parent = 0 ); | 167 | OPrismHeaderPacket( const unsigned char*, const struct prism_hdr*, QObject* parent = 0 ); |
@@ -479,128 +481,139 @@ class OUDPPacket : public QObject | |||
479 | private: | 481 | private: |
480 | const struct udphdr* _udphdr; | 482 | const struct udphdr* _udphdr; |
481 | }; | 483 | }; |
482 | 484 | ||
483 | /*====================================================================================== | 485 | /*====================================================================================== |
484 | * OTCPPacket | 486 | * OTCPPacket |
485 | *======================================================================================*/ | 487 | *======================================================================================*/ |
486 | 488 | ||
487 | class OTCPPacket : public QObject | 489 | class OTCPPacket : public QObject |
488 | { | 490 | { |
489 | Q_OBJECT | 491 | Q_OBJECT |
490 | 492 | ||
491 | public: | 493 | public: |
492 | OTCPPacket( const unsigned char*, const struct tcphdr*, QObject* parent = 0 ); | 494 | OTCPPacket( const unsigned char*, const struct tcphdr*, QObject* parent = 0 ); |
493 | virtual ~OTCPPacket(); | 495 | virtual ~OTCPPacket(); |
494 | 496 | ||
495 | int fromPort() const; | 497 | int fromPort() const; |
496 | int toPort() const; | 498 | int toPort() const; |
497 | 499 | ||
498 | private: | 500 | private: |
499 | const struct tcphdr* _tcphdr; | 501 | const struct tcphdr* _tcphdr; |
500 | }; | 502 | }; |
501 | 503 | ||
502 | 504 | ||
503 | /*====================================================================================== | 505 | /*====================================================================================== |
504 | * OPacketCapturer | 506 | * OPacketCapturer |
505 | *======================================================================================*/ | 507 | *======================================================================================*/ |
506 | 508 | ||
507 | /** | 509 | /** |
508 | * @brief A class based wrapper for network packet capturing. | 510 | * @brief A class based wrapper for network packet capturing. |
509 | * | 511 | * |
510 | * This class is the base of a high-level interface to the well known packet capturing | 512 | * This class is the base of a high-level interface to the well known packet capturing |
511 | * library libpcap. ... | 513 | * library libpcap. ... |
512 | */ | 514 | */ |
513 | class OPacketCapturer : public QObject | 515 | class OPacketCapturer : public QObject |
514 | { | 516 | { |
515 | Q_OBJECT | 517 | Q_OBJECT |
516 | 518 | ||
517 | public: | 519 | public: |
518 | /** | 520 | /** |
519 | * Constructor. | 521 | * Constructor. |
520 | */ | 522 | */ |
521 | OPacketCapturer( QObject* parent = 0, const char* name = 0 ); | 523 | OPacketCapturer( QObject* parent = 0, const char* name = 0 ); |
522 | /** | 524 | /** |
523 | * Destructor. | 525 | * Destructor. |
524 | */ | 526 | */ |
525 | ~OPacketCapturer(); | 527 | ~OPacketCapturer(); |
526 | /** | 528 | /** |
527 | * Setting the packet capturer to use blocking IO calls can be useful when | 529 | * Set the packet capturer to use blocking or non-blocking IO. This can be useful when |
528 | * not using the socket notifier, e.g. without an application object. | 530 | * not using the socket notifier, e.g. without an application object. |
529 | */ | 531 | */ |
530 | void setBlocking( bool ); | 532 | void setBlocking( bool ); |
531 | /** | 533 | /** |
532 | * @returns true if the packet capturer uses blocking IO calls. | 534 | * @returns true if the packet capturer uses blocking IO calls. |
533 | */ | 535 | */ |
534 | bool blocking() const; | 536 | bool blocking() const; |
535 | /** | 537 | /** |
536 | * Closes the packet capturer. This is automatically done in the destructor. | 538 | * Close the packet capturer. This is automatically done in the destructor. |
537 | */ | 539 | */ |
538 | void close(); | 540 | void close(); |
539 | /** | 541 | /** |
542 | * Close the output capture file. | ||
543 | */ | ||
544 | void closeDumpFile(); | ||
545 | /** | ||
540 | * @returns the data link type. | 546 | * @returns the data link type. |
541 | * @see <pcap.h> for possible values. | 547 | * @see <pcap.h> for possible values. |
542 | */ | 548 | */ |
543 | int dataLink() const; | 549 | int dataLink() const; |
544 | /** | 550 | /** |
551 | * Dump a packet to the output capture file. | ||
552 | */ | ||
553 | void dump( OPacket* ); | ||
554 | /** | ||
545 | * @returns the file descriptor of the packet capturer. This is only useful, if | 555 | * @returns the file descriptor of the packet capturer. This is only useful, if |
546 | * not using the socket notifier, e.g. without an application object. | 556 | * not using the socket notifier, e.g. without an application object. |
547 | */ | 557 | */ |
548 | int fileno() const; | 558 | int fileno() const; |
549 | /** | 559 | /** |
550 | * @returns the next @ref OPacket from the packet capturer. | 560 | * @returns the next @ref OPacket from the packet capturer. |
551 | * @note If blocking mode is true then this call might block. | 561 | * @note If blocking mode is true then this call might block. |
552 | */ | 562 | */ |
553 | OPacket* next(); | 563 | OPacket* next(); |
554 | /** | 564 | /** |
555 | * Open the packet capturer to capture packets in live-mode from @a interface. | 565 | * Open the packet capturer to capture packets in live-mode from @a interface. |
556 | * If a @a filename is given, all captured packets are output to a tcpdump-compatible capture file. | ||
557 | */ | 566 | */ |
558 | bool open( const QString& interface, const QString& filename = QString::null ); | 567 | bool open( const QString& interface ); |
559 | /** | 568 | /** |
560 | * Open the packet capturer to capture packets in offline-mode from @a file. | 569 | * Open the packet capturer to capture packets in offline-mode from @a file. |
561 | */ | 570 | */ |
562 | bool open( const QFile& file ); | 571 | bool open( const QFile& file ); |
563 | /** | 572 | /** |
573 | * Open a prerecorded tcpdump compatible capture file for use with @ref dump() | ||
574 | */ | ||
575 | bool openDumpFile( const QString& filename ); | ||
576 | /** | ||
564 | * @returns true if the packet capturer is open | 577 | * @returns true if the packet capturer is open |
565 | */ | 578 | */ |
566 | bool isOpen() const; | 579 | bool isOpen() const; |
567 | /** | 580 | /** |
568 | * @returns the snapshot length of this packet capturer | 581 | * @returns the snapshot length of this packet capturer |
569 | */ | 582 | */ |
570 | int snapShot() const; | 583 | int snapShot() const; |
571 | /** | 584 | /** |
572 | * @returns true if the input capture file has a different byte-order | 585 | * @returns true if the input capture file has a different byte-order |
573 | * than the byte-order of the running system. | 586 | * than the byte-order of the running system. |
574 | */ | 587 | */ |
575 | bool swapped() const; | 588 | bool swapped() const; |
576 | /** | 589 | /** |
577 | * @returns the libpcap version string used to write the input capture file. | 590 | * @returns the libpcap version string used to write the input capture file. |
578 | */ | 591 | */ |
579 | QString version() const; | 592 | QString version() const; |
580 | /** | 593 | /** |
581 | * @returns the packet statistic database. | 594 | * @returns the packet statistic database. |
582 | * @see QMap | 595 | * @see QMap |
583 | */ | 596 | */ |
584 | const QMap<QString,int>& statistics() const; | 597 | const QMap<QString,int>& statistics() const; |
585 | 598 | ||
586 | signals: | 599 | signals: |
587 | /** | 600 | /** |
588 | * This signal is emitted, when a packet has been received. | 601 | * This signal is emitted, when a packet has been received. |
589 | */ | 602 | */ |
590 | void receivedPacket( OPacket* ); | 603 | void receivedPacket( OPacket* ); |
591 | 604 | ||
592 | protected slots: | 605 | protected slots: |
593 | void readyToReceive(); | 606 | void readyToReceive(); |
594 | 607 | ||
595 | protected: | 608 | protected: |
596 | QString _name; // devicename | 609 | QString _name; // devicename |
597 | bool _open; // check this before doing pcap calls | 610 | bool _open; // check this before doing pcap calls |
598 | pcap_t* _pch; // pcap library handle | 611 | pcap_t* _pch; // pcap library handle |
599 | pcap_dumper_t* _pcd; // pcap dumper handle | 612 | pcap_dumper_t* _pcd; // pcap dumper handle |
600 | QSocketNotifier* _sn; // socket notifier for main loop | 613 | QSocketNotifier* _sn; // socket notifier for main loop |
601 | mutable char _errbuf[PCAP_ERRBUF_SIZE]; // holds error strings from libpcap | 614 | mutable char _errbuf[PCAP_ERRBUF_SIZE]; // holds error strings from libpcap |
602 | QMap<QString, int> _stats; // statistics; | 615 | QMap<QString, int> _stats; // statistics; |
603 | }; | 616 | }; |
604 | 617 | ||
605 | #endif // OPCAP_H | 618 | #endif // OPCAP_H |
606 | 619 | ||