author | mickeyl <mickeyl> | 2003-04-16 13:19:32 (UTC) |
---|---|---|
committer | mickeyl <mickeyl> | 2003-04-16 13:19:32 (UTC) |
commit | 05b76911ab2082436c577c1461f0d1210ce0aa33 (patch) (unidiff) | |
tree | d8219eb138ca46f355651152d471664037be372b /libopie2 | |
parent | 5b9d1ddde859ff783f95babf1887fa40e6bfe0be (diff) | |
download | opie-05b76911ab2082436c577c1461f0d1210ce0aa33.zip opie-05b76911ab2082436c577c1461f0d1210ce0aa33.tar.gz opie-05b76911ab2082436c577c1461f0d1210ce0aa33.tar.bz2 |
add sanity check for last packet when capturing from capture file
-rw-r--r-- | libopie2/opienet/opcap.cpp | 3 |
1 files changed, 2 insertions, 1 deletions
diff --git a/libopie2/opienet/opcap.cpp b/libopie2/opienet/opcap.cpp index 04b1bb1..e362883 100644 --- a/libopie2/opienet/opcap.cpp +++ b/libopie2/opienet/opcap.cpp | |||
@@ -639,328 +639,329 @@ OWaveLanManagementIBSS::OWaveLanManagementIBSS( const unsigned char* end, const | |||
639 | { | 639 | { |
640 | qDebug( "OWaveLanManagementIBSS()" ); | 640 | qDebug( "OWaveLanManagementIBSS()" ); |
641 | } | 641 | } |
642 | 642 | ||
643 | 643 | ||
644 | OWaveLanManagementIBSS::~OWaveLanManagementIBSS() | 644 | OWaveLanManagementIBSS::~OWaveLanManagementIBSS() |
645 | { | 645 | { |
646 | } | 646 | } |
647 | 647 | ||
648 | /*====================================================================================== | 648 | /*====================================================================================== |
649 | * OWaveLanManagementChallenge | 649 | * OWaveLanManagementChallenge |
650 | *======================================================================================*/ | 650 | *======================================================================================*/ |
651 | 651 | ||
652 | OWaveLanManagementChallenge::OWaveLanManagementChallenge( const unsigned char* end, const struct challenge_t* data, QObject* parent ) | 652 | OWaveLanManagementChallenge::OWaveLanManagementChallenge( const unsigned char* end, const struct challenge_t* data, QObject* parent ) |
653 | :QObject( parent, "802.11 Challenge" ), _data( data ) | 653 | :QObject( parent, "802.11 Challenge" ), _data( data ) |
654 | { | 654 | { |
655 | qDebug( "OWaveLanManagementChallenge()" ); | 655 | qDebug( "OWaveLanManagementChallenge()" ); |
656 | } | 656 | } |
657 | 657 | ||
658 | 658 | ||
659 | OWaveLanManagementChallenge::~OWaveLanManagementChallenge() | 659 | OWaveLanManagementChallenge::~OWaveLanManagementChallenge() |
660 | { | 660 | { |
661 | } | 661 | } |
662 | 662 | ||
663 | /*====================================================================================== | 663 | /*====================================================================================== |
664 | * OWaveLanDataPacket | 664 | * OWaveLanDataPacket |
665 | *======================================================================================*/ | 665 | *======================================================================================*/ |
666 | 666 | ||
667 | OWaveLanDataPacket::OWaveLanDataPacket( const unsigned char* end, const struct ieee_802_11_data_header* data, OWaveLanPacket* parent ) | 667 | OWaveLanDataPacket::OWaveLanDataPacket( const unsigned char* end, const struct ieee_802_11_data_header* data, OWaveLanPacket* parent ) |
668 | :QObject( parent, "802.11 Data" ), _header( data ) | 668 | :QObject( parent, "802.11 Data" ), _header( data ) |
669 | { | 669 | { |
670 | qDebug( "OWaveLanDataPacket::OWaveLanDataPacket(): decoding frame..." ); | 670 | qDebug( "OWaveLanDataPacket::OWaveLanDataPacket(): decoding frame..." ); |
671 | 671 | ||
672 | const unsigned char* payload = (const unsigned char*) data + sizeof( struct ieee_802_11_data_header ); | 672 | const unsigned char* payload = (const unsigned char*) data + sizeof( struct ieee_802_11_data_header ); |
673 | 673 | ||
674 | #warning The next line works for most cases, but can not be correct generally! | 674 | #warning The next line works for most cases, but can not be correct generally! |
675 | if (!( ( (OWaveLanPacket*) this->parent())->duration() )) payload -= 6; // compensation for missing last address | 675 | if (!( ( (OWaveLanPacket*) this->parent())->duration() )) payload -= 6; // compensation for missing last address |
676 | 676 | ||
677 | new OLLCPacket( end, (const struct ieee_802_11_802_2_header*) payload, this ); | 677 | new OLLCPacket( end, (const struct ieee_802_11_802_2_header*) payload, this ); |
678 | } | 678 | } |
679 | 679 | ||
680 | 680 | ||
681 | OWaveLanDataPacket::~OWaveLanDataPacket() | 681 | OWaveLanDataPacket::~OWaveLanDataPacket() |
682 | { | 682 | { |
683 | } | 683 | } |
684 | 684 | ||
685 | 685 | ||
686 | /*====================================================================================== | 686 | /*====================================================================================== |
687 | * OLLCPacket | 687 | * OLLCPacket |
688 | *======================================================================================*/ | 688 | *======================================================================================*/ |
689 | 689 | ||
690 | OLLCPacket::OLLCPacket( const unsigned char* end, const struct ieee_802_11_802_2_header* data, QObject* parent ) | 690 | OLLCPacket::OLLCPacket( const unsigned char* end, const struct ieee_802_11_802_2_header* data, QObject* parent ) |
691 | :QObject( parent, "802.11 LLC" ), _header( data ) | 691 | :QObject( parent, "802.11 LLC" ), _header( data ) |
692 | { | 692 | { |
693 | qDebug( "OLLCPacket::OLLCPacket(): decoding frame..." ); | 693 | qDebug( "OLLCPacket::OLLCPacket(): decoding frame..." ); |
694 | 694 | ||
695 | if ( !(_header->oui[0] || _header->oui[1] || _header->oui[2]) ) | 695 | if ( !(_header->oui[0] || _header->oui[1] || _header->oui[2]) ) |
696 | { | 696 | { |
697 | qDebug( "OLLCPacket::OLLCPacket(): contains an encapsulated Ethernet frame (type=%04X)", EXTRACT_16BITS( &_header->type ) ); | 697 | qDebug( "OLLCPacket::OLLCPacket(): contains an encapsulated Ethernet frame (type=%04X)", EXTRACT_16BITS( &_header->type ) ); |
698 | 698 | ||
699 | switch ( EXTRACT_16BITS( &_header->type ) ) // defined in linux/if_ether.h | 699 | switch ( EXTRACT_16BITS( &_header->type ) ) // defined in linux/if_ether.h |
700 | { | 700 | { |
701 | case ETH_P_IP: new OIPPacket( end, (const struct iphdr*) (data+1), this ); break; | 701 | case ETH_P_IP: new OIPPacket( end, (const struct iphdr*) (data+1), this ); break; |
702 | default: qDebug( "OLLCPacket::OLLCPacket(): Unknown Encapsulation Type" ); | 702 | default: qDebug( "OLLCPacket::OLLCPacket(): Unknown Encapsulation Type" ); |
703 | } | 703 | } |
704 | 704 | ||
705 | } | 705 | } |
706 | } | 706 | } |
707 | 707 | ||
708 | 708 | ||
709 | OLLCPacket::~OLLCPacket() | 709 | OLLCPacket::~OLLCPacket() |
710 | { | 710 | { |
711 | } | 711 | } |
712 | 712 | ||
713 | 713 | ||
714 | /*====================================================================================== | 714 | /*====================================================================================== |
715 | * OWaveLanControlPacket | 715 | * OWaveLanControlPacket |
716 | *======================================================================================*/ | 716 | *======================================================================================*/ |
717 | 717 | ||
718 | OWaveLanControlPacket::OWaveLanControlPacket( const unsigned char* end, const struct ieee_802_11_control_header* data, OWaveLanPacket* parent ) | 718 | OWaveLanControlPacket::OWaveLanControlPacket( const unsigned char* end, const struct ieee_802_11_control_header* data, OWaveLanPacket* parent ) |
719 | :QObject( parent, "802.11 Data" ), _header( data ) | 719 | :QObject( parent, "802.11 Data" ), _header( data ) |
720 | { | 720 | { |
721 | qDebug( "OWaveLanControlPacket::OWaveLanDataControl(): decoding frame..." ); | 721 | qDebug( "OWaveLanControlPacket::OWaveLanDataControl(): decoding frame..." ); |
722 | //TODO: Implement this | 722 | //TODO: Implement this |
723 | } | 723 | } |
724 | 724 | ||
725 | 725 | ||
726 | OWaveLanControlPacket::~OWaveLanControlPacket() | 726 | OWaveLanControlPacket::~OWaveLanControlPacket() |
727 | { | 727 | { |
728 | } | 728 | } |
729 | 729 | ||
730 | 730 | ||
731 | /*====================================================================================== | 731 | /*====================================================================================== |
732 | * OPacketCapturer | 732 | * OPacketCapturer |
733 | *======================================================================================*/ | 733 | *======================================================================================*/ |
734 | 734 | ||
735 | OPacketCapturer::OPacketCapturer( QObject* parent, const char* name ) | 735 | OPacketCapturer::OPacketCapturer( QObject* parent, const char* name ) |
736 | :QObject( parent, name ), _name( QString::null ), _open( false ), | 736 | :QObject( parent, name ), _name( QString::null ), _open( false ), |
737 | _pch( 0 ), _pcd( 0 ), _sn( 0 ) | 737 | _pch( 0 ), _pcd( 0 ), _sn( 0 ) |
738 | { | 738 | { |
739 | } | 739 | } |
740 | 740 | ||
741 | 741 | ||
742 | OPacketCapturer::~OPacketCapturer() | 742 | OPacketCapturer::~OPacketCapturer() |
743 | { | 743 | { |
744 | if ( _open ) | 744 | if ( _open ) |
745 | { | 745 | { |
746 | qDebug( "OPacketCapturer::~OPacketCapturer(): pcap still open, autoclosing." ); | 746 | qDebug( "OPacketCapturer::~OPacketCapturer(): pcap still open, autoclosing." ); |
747 | close(); | 747 | close(); |
748 | } | 748 | } |
749 | } | 749 | } |
750 | 750 | ||
751 | 751 | ||
752 | void OPacketCapturer::setBlocking( bool b ) | 752 | void OPacketCapturer::setBlocking( bool b ) |
753 | { | 753 | { |
754 | if ( pcap_setnonblock( _pch, 1-b, _errbuf ) != -1 ) | 754 | if ( pcap_setnonblock( _pch, 1-b, _errbuf ) != -1 ) |
755 | { | 755 | { |
756 | qDebug( "OPacketCapturer::setBlocking(): blocking mode changed successfully." ); | 756 | qDebug( "OPacketCapturer::setBlocking(): blocking mode changed successfully." ); |
757 | } | 757 | } |
758 | else | 758 | else |
759 | { | 759 | { |
760 | qDebug( "OPacketCapturer::setBlocking(): can't change blocking mode: %s", _errbuf ); | 760 | qDebug( "OPacketCapturer::setBlocking(): can't change blocking mode: %s", _errbuf ); |
761 | } | 761 | } |
762 | } | 762 | } |
763 | 763 | ||
764 | 764 | ||
765 | bool OPacketCapturer::blocking() const | 765 | bool OPacketCapturer::blocking() const |
766 | { | 766 | { |
767 | int b = pcap_getnonblock( _pch, _errbuf ); | 767 | int b = pcap_getnonblock( _pch, _errbuf ); |
768 | if ( b == -1 ) | 768 | if ( b == -1 ) |
769 | { | 769 | { |
770 | qDebug( "OPacketCapturer::blocking(): can't get blocking mode: %s", _errbuf ); | 770 | qDebug( "OPacketCapturer::blocking(): can't get blocking mode: %s", _errbuf ); |
771 | return -1; | 771 | return -1; |
772 | } | 772 | } |
773 | return !b; | 773 | return !b; |
774 | } | 774 | } |
775 | 775 | ||
776 | 776 | ||
777 | void OPacketCapturer::close() | 777 | void OPacketCapturer::close() |
778 | { | 778 | { |
779 | if ( _open ) | 779 | if ( _open ) |
780 | { | 780 | { |
781 | if ( _sn ) | 781 | if ( _sn ) |
782 | { | 782 | { |
783 | _sn->disconnect( SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); | 783 | _sn->disconnect( SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); |
784 | delete _sn; | 784 | delete _sn; |
785 | } | 785 | } |
786 | if ( _pcd ) | 786 | if ( _pcd ) |
787 | { | 787 | { |
788 | pcap_dump_close( _pcd ); | 788 | pcap_dump_close( _pcd ); |
789 | _pcd = 0; | 789 | _pcd = 0; |
790 | } | 790 | } |
791 | pcap_close( _pch ); | 791 | pcap_close( _pch ); |
792 | _open = false; | 792 | _open = false; |
793 | } | 793 | } |
794 | 794 | ||
795 | qDebug( "OPacketCapturer::close() --- dumping capturing statistics..." ); | 795 | qDebug( "OPacketCapturer::close() --- dumping capturing statistics..." ); |
796 | qDebug( "--------------------------------------------------" ); | 796 | qDebug( "--------------------------------------------------" ); |
797 | for( QMap<QString,int>::Iterator it = _stats.begin(); it != _stats.end(); ++it ) | 797 | for( QMap<QString,int>::Iterator it = _stats.begin(); it != _stats.end(); ++it ) |
798 | qDebug( "%s : %d", (const char*) it.key(), it.data() ); | 798 | qDebug( "%s : %d", (const char*) it.key(), it.data() ); |
799 | qDebug( "--------------------------------------------------" ); | 799 | qDebug( "--------------------------------------------------" ); |
800 | 800 | ||
801 | } | 801 | } |
802 | 802 | ||
803 | 803 | ||
804 | int OPacketCapturer::dataLink() const | 804 | int OPacketCapturer::dataLink() const |
805 | { | 805 | { |
806 | return pcap_datalink( _pch ); | 806 | return pcap_datalink( _pch ); |
807 | } | 807 | } |
808 | 808 | ||
809 | 809 | ||
810 | int OPacketCapturer::fileno() const | 810 | int OPacketCapturer::fileno() const |
811 | { | 811 | { |
812 | if ( _open ) | 812 | if ( _open ) |
813 | { | 813 | { |
814 | return pcap_fileno( _pch ); | 814 | return pcap_fileno( _pch ); |
815 | } | 815 | } |
816 | else | 816 | else |
817 | { | 817 | { |
818 | return -1; | 818 | return -1; |
819 | } | 819 | } |
820 | } | 820 | } |
821 | 821 | ||
822 | OPacket* OPacketCapturer::next() | 822 | OPacket* OPacketCapturer::next() |
823 | { | 823 | { |
824 | packetheaderstruct header; | 824 | packetheaderstruct header; |
825 | qDebug( "==> OPacketCapturer::next()" ); | 825 | qDebug( "==> OPacketCapturer::next()" ); |
826 | const unsigned char* pdata = pcap_next( _pch, &header ); | 826 | const unsigned char* pdata = pcap_next( _pch, &header ); |
827 | qDebug( "<== OPacketCapturer::next()" ); | 827 | qDebug( "<== OPacketCapturer::next()" ); |
828 | if ( _pcd ) | 828 | if ( _pcd ) |
829 | pcap_dump( (u_char*) _pcd, &header, pdata ); | 829 | pcap_dump( (u_char*) _pcd, &header, pdata ); |
830 | 830 | ||
831 | if ( header.len ) | 831 | if ( pdata && header.len ) |
832 | { | 832 | { |
833 | OPacket* p = new OPacket( dataLink(), header, pdata, 0 ); | 833 | OPacket* p = new OPacket( dataLink(), header, pdata, 0 ); |
834 | // packets shouldn't be inserted in the QObject child-parent hierarchy, | 834 | // packets shouldn't be inserted in the QObject child-parent hierarchy, |
835 | // because due to memory constraints they will be deleted as soon | 835 | // because due to memory constraints they will be deleted as soon |
836 | // as possible - that is right after they have been processed | 836 | // as possible - that is right after they have been processed |
837 | // by emit() [ see below ] | 837 | // by emit() [ see below ] |
838 | //TODO: make gathering statistics optional, because it takes time | 838 | //TODO: make gathering statistics optional, because it takes time |
839 | p->updateStats( _stats, const_cast<QObjectList*>( p->children() ) ); | 839 | p->updateStats( _stats, const_cast<QObjectList*>( p->children() ) ); |
840 | 840 | ||
841 | return p; | 841 | return p; |
842 | } | 842 | } |
843 | else | 843 | else |
844 | { | 844 | { |
845 | qWarning( "OPacketCapturer::next() - no packet received!" ); | ||
845 | return 0; | 846 | return 0; |
846 | } | 847 | } |
847 | } | 848 | } |
848 | 849 | ||
849 | 850 | ||
850 | bool OPacketCapturer::open( const QString& name, const QString& filename ) | 851 | bool OPacketCapturer::open( const QString& name, const QString& filename ) |
851 | { | 852 | { |
852 | if ( _open ) | 853 | if ( _open ) |
853 | { | 854 | { |
854 | if ( name == _name ) // ignore opening an already openend device | 855 | if ( name == _name ) // ignore opening an already openend device |
855 | { | 856 | { |
856 | return true; | 857 | return true; |
857 | } | 858 | } |
858 | else // close the last opened device | 859 | else // close the last opened device |
859 | { | 860 | { |
860 | close(); | 861 | close(); |
861 | } | 862 | } |
862 | } | 863 | } |
863 | 864 | ||
864 | _name = name; | 865 | _name = name; |
865 | 866 | ||
866 | // open libpcap | 867 | // open libpcap |
867 | pcap_t* handle = pcap_open_live( const_cast<char*>( (const char*) name ), 1024, 0, 0, &_errbuf[0] ); | 868 | pcap_t* handle = pcap_open_live( const_cast<char*>( (const char*) name ), 1024, 0, 0, &_errbuf[0] ); |
868 | 869 | ||
869 | if ( !handle ) | 870 | if ( !handle ) |
870 | { | 871 | { |
871 | qWarning( "OPacketCapturer::open(): can't open libpcap with '%s': %s", (const char*) name, _errbuf ); | 872 | qWarning( "OPacketCapturer::open(): can't open libpcap with '%s': %s", (const char*) name, _errbuf ); |
872 | return false; | 873 | return false; |
873 | } | 874 | } |
874 | 875 | ||
875 | qDebug( "OPacketCapturer::open(): libpcap [%s] opened successfully.", (const char*) name ); | 876 | qDebug( "OPacketCapturer::open(): libpcap [%s] opened successfully.", (const char*) name ); |
876 | _pch = handle; | 877 | _pch = handle; |
877 | _open = true; | 878 | _open = true; |
878 | _stats.clear(); | 879 | _stats.clear(); |
879 | 880 | ||
880 | // in case we have an application object, create a socket notifier | 881 | // in case we have an application object, create a socket notifier |
881 | if ( qApp ) //TODO: I don't like this here... | 882 | if ( qApp ) //TODO: I don't like this here... |
882 | { | 883 | { |
883 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); | 884 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); |
884 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); | 885 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); |
885 | } | 886 | } |
886 | 887 | ||
887 | // if requested, open a dump | 888 | // if requested, open a dump |
888 | pcap_dumper_t* dump = pcap_dump_open( _pch, const_cast<char*>( (const char*) filename ) ); | 889 | pcap_dumper_t* dump = pcap_dump_open( _pch, const_cast<char*>( (const char*) filename ) ); |
889 | if ( !dump ) | 890 | if ( !dump ) |
890 | { | 891 | { |
891 | qWarning( "OPacketCapturer::open(): can't open dump with '%s': %s", (const char*) filename, _errbuf ); | 892 | qWarning( "OPacketCapturer::open(): can't open dump with '%s': %s", (const char*) filename, _errbuf ); |
892 | return false; | 893 | return false; |
893 | } | 894 | } |
894 | qDebug( "OPacketCapturer::open(): dump [%s] opened successfully.", (const char*) filename ); | 895 | qDebug( "OPacketCapturer::open(): dump [%s] opened successfully.", (const char*) filename ); |
895 | _pcd = dump; | 896 | _pcd = dump; |
896 | 897 | ||
897 | return true; | 898 | return true; |
898 | } | 899 | } |
899 | 900 | ||
900 | 901 | ||
901 | bool OPacketCapturer::open( const QFile& file ) | 902 | bool OPacketCapturer::open( const QFile& file ) |
902 | { | 903 | { |
903 | QString name = file.name(); | 904 | QString name = file.name(); |
904 | 905 | ||
905 | if ( _open ) | 906 | if ( _open ) |
906 | { | 907 | { |
907 | close(); | 908 | close(); |
908 | if ( name == _name ) // ignore opening an already openend device | 909 | if ( name == _name ) // ignore opening an already openend device |
909 | { | 910 | { |
910 | return true; | 911 | return true; |
911 | } | 912 | } |
912 | else // close the last opened device | 913 | else // close the last opened device |
913 | { | 914 | { |
914 | close(); | 915 | close(); |
915 | } | 916 | } |
916 | } | 917 | } |
917 | 918 | ||
918 | _name = name; | 919 | _name = name; |
919 | 920 | ||
920 | pcap_t* handle = pcap_open_offline( const_cast<char*>( (const char*) name ), &_errbuf[0] ); | 921 | pcap_t* handle = pcap_open_offline( const_cast<char*>( (const char*) name ), &_errbuf[0] ); |
921 | 922 | ||
922 | if ( handle ) | 923 | if ( handle ) |
923 | { | 924 | { |
924 | qDebug( "OPacketCapturer::open(): libpcap opened successfully." ); | 925 | qDebug( "OPacketCapturer::open(): libpcap opened successfully." ); |
925 | _pch = handle; | 926 | _pch = handle; |
926 | _open = true; | 927 | _open = true; |
927 | 928 | ||
928 | // in case we have an application object, create a socket notifier | 929 | // in case we have an application object, create a socket notifier |
929 | if ( qApp ) | 930 | if ( qApp ) |
930 | { | 931 | { |
931 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); | 932 | _sn = new QSocketNotifier( fileno(), QSocketNotifier::Read ); |
932 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); | 933 | connect( _sn, SIGNAL( activated(int) ), this, SLOT( readyToReceive() ) ); |
933 | } | 934 | } |
934 | 935 | ||
935 | return true; | 936 | return true; |
936 | } | 937 | } |
937 | else | 938 | else |
938 | { | 939 | { |
939 | qDebug( "OPacketCapturer::open(): can't open libpcap with '%s': %s", (const char*) name, _errbuf ); | 940 | qDebug( "OPacketCapturer::open(): can't open libpcap with '%s': %s", (const char*) name, _errbuf ); |
940 | return false; | 941 | return false; |
941 | } | 942 | } |
942 | 943 | ||
943 | } | 944 | } |
944 | 945 | ||
945 | 946 | ||
946 | bool OPacketCapturer::isOpen() const | 947 | bool OPacketCapturer::isOpen() const |
947 | { | 948 | { |
948 | return _open; | 949 | return _open; |
949 | } | 950 | } |
950 | 951 | ||
951 | 952 | ||
952 | void OPacketCapturer::readyToReceive() | 953 | void OPacketCapturer::readyToReceive() |
953 | { | 954 | { |
954 | qDebug( "OPacketCapturer::readyToReceive(): about to emit 'receivePacket(p)'" ); | 955 | qDebug( "OPacketCapturer::readyToReceive(): about to emit 'receivePacket(p)'" ); |
955 | OPacket* p = next(); | 956 | OPacket* p = next(); |
956 | emit receivedPacket( p ); | 957 | emit receivedPacket( p ); |
957 | // emit is synchronous - packet has been dealt with, now it's safe to delete | 958 | // emit is synchronous - packet has been dealt with, now it's safe to delete |
958 | delete p; | 959 | delete p; |
959 | } | 960 | } |
960 | 961 | ||
961 | 962 | ||
962 | const QMap<QString,int>& OPacketCapturer::statistics() const | 963 | const QMap<QString,int>& OPacketCapturer::statistics() const |
963 | { | 964 | { |
964 | return _stats; | 965 | return _stats; |
965 | } | 966 | } |
966 | 967 | ||